roojs-ui.js
[roojs1] / roojs-debug.js
1 /*
2  * Based on:
3  * Ext JS Library 1.1.1
4  * Copyright(c) 2006-2007, Ext JS, LLC.
5  *
6  * Originally Released Under LGPL - original licence link has changed is not relivant.
7  *
8  * Fork - LGPL
9  * <script type="text/javascript">
10  */
11  
12
13
14
15
16 // for old browsers
17 window["undefined"] = window["undefined"];
18
19 /**
20  * @class Roo
21  * Roo core utilities and functions.
22  * @singleton
23  */
24 var Roo = {}; 
25 /**
26  * Copies all the properties of config to obj.
27  * @param {Object} obj The receiver of the properties
28  * @param {Object} config The source of the properties
29  * @param {Object} defaults A different object that will also be applied for default values
30  * @return {Object} returns obj
31  * @member Roo apply
32  */
33
34  
35 Roo.apply = function(o, c, defaults){
36     if(defaults){
37         // no "this" reference for friendly out of scope calls
38         Roo.apply(o, defaults);
39     }
40     if(o && c && typeof c == 'object'){
41         for(var p in c){
42             o[p] = c[p];
43         }
44     }
45     return o;
46 };
47
48
49 (function(){
50     var idSeed = 0;
51     var ua = navigator.userAgent.toLowerCase();
52
53     var isStrict = document.compatMode == "CSS1Compat",
54         isOpera = ua.indexOf("opera") > -1,
55         isSafari = (/webkit|khtml/).test(ua),
56         isFirefox = ua.indexOf("firefox") > -1,
57         isIE = ua.indexOf("msie") > -1,
58         isIE7 = ua.indexOf("msie 7") > -1,
59         isIE11 = /trident.*rv\:11\./.test(ua),
60         isEdge = ua.indexOf("edge") > -1,
61         isGecko = !isSafari && ua.indexOf("gecko") > -1,
62         isBorderBox = isIE && !isStrict,
63         isWindows = (ua.indexOf("windows") != -1 || ua.indexOf("win32") != -1),
64         isMac = (ua.indexOf("macintosh") != -1 || ua.indexOf("mac os x") != -1),
65         isLinux = (ua.indexOf("linux") != -1),
66         isSecure = window.location.href.toLowerCase().indexOf("https") === 0,
67         isIOS = /iphone|ipad/.test(ua),
68         isAndroid = /android/.test(ua),
69         isTouch =  (function() {
70             try {
71                 if (ua.indexOf('chrome') != -1 && ua.indexOf('android') == -1) {
72                     window.addEventListener('touchstart', function __set_has_touch__ () {
73                         Roo.isTouch = true;
74                         window.removeEventListener('touchstart', __set_has_touch__);
75                     });
76                     return false; // no touch on chrome!?
77                 }
78                 document.createEvent("TouchEvent");  
79                 return true;  
80             } catch (e) {  
81                 return false;  
82             } 
83             
84         })();
85     // remove css image flicker
86         if(isIE && !isIE7){
87         try{
88             document.execCommand("BackgroundImageCache", false, true);
89         }catch(e){}
90     }
91     
92     Roo.apply(Roo, {
93         /**
94          * True if the browser is in strict mode
95          * @type Boolean
96          */
97         isStrict : isStrict,
98         /**
99          * True if the page is running over SSL
100          * @type Boolean
101          */
102         isSecure : isSecure,
103         /**
104          * True when the document is fully initialized and ready for action
105          * @type Boolean
106          */
107         isReady : false,
108         /**
109          * Turn on debugging output (currently only the factory uses this)
110          * @type Boolean
111          */
112         
113         debug: false,
114
115         /**
116          * True to automatically uncache orphaned Roo.Elements periodically (defaults to true)
117          * @type Boolean
118          */
119         enableGarbageCollector : true,
120
121         /**
122          * True to automatically purge event listeners after uncaching an element (defaults to false).
123          * Note: this only happens if enableGarbageCollector is true.
124          * @type Boolean
125          */
126         enableListenerCollection:false,
127
128         /**
129          * URL to a blank file used by Roo when in secure mode for iframe src and onReady src to prevent
130          * the IE insecure content warning (defaults to javascript:false).
131          * @type String
132          */
133         SSL_SECURE_URL : "javascript:false",
134
135         /**
136          * URL to a 1x1 transparent gif image used by Roo to create inline icons with CSS background images. (Defaults to
137          * "http://Roojs.com/s.gif" and you should change this to a URL on your server).
138          * @type String
139          */
140         BLANK_IMAGE_URL : "http:/"+"/localhost/s.gif",
141
142         emptyFn : function(){},
143         
144         /**
145          * Copies all the properties of config to obj if they don't already exist.
146          * @param {Object} obj The receiver of the properties
147          * @param {Object} config The source of the properties
148          * @return {Object} returns obj
149          */
150         applyIf : function(o, c){
151             if(o && c){
152                 for(var p in c){
153                     if(typeof o[p] == "undefined"){ o[p] = c[p]; }
154                 }
155             }
156             return o;
157         },
158
159         /**
160          * Applies event listeners to elements by selectors when the document is ready.
161          * The event name is specified with an @ suffix.
162 <pre><code>
163 Roo.addBehaviors({
164    // add a listener for click on all anchors in element with id foo
165    '#foo a@click' : function(e, t){
166        // do something
167    },
168
169    // add the same listener to multiple selectors (separated by comma BEFORE the @)
170    '#foo a, #bar span.some-class@mouseover' : function(){
171        // do something
172    }
173 });
174 </code></pre>
175          * @param {Object} obj The list of behaviors to apply
176          */
177         addBehaviors : function(o){
178             if(!Roo.isReady){
179                 Roo.onReady(function(){
180                     Roo.addBehaviors(o);
181                 });
182                 return;
183             }
184             var cache = {}; // simple cache for applying multiple behaviors to same selector does query multiple times
185             for(var b in o){
186                 var parts = b.split('@');
187                 if(parts[1]){ // for Object prototype breakers
188                     var s = parts[0];
189                     if(!cache[s]){
190                         cache[s] = Roo.select(s);
191                     }
192                     cache[s].on(parts[1], o[b]);
193                 }
194             }
195             cache = null;
196         },
197
198         /**
199          * Generates unique ids. If the element already has an id, it is unchanged
200          * @param {String/HTMLElement/Element} el (optional) The element to generate an id for
201          * @param {String} prefix (optional) Id prefix (defaults "Roo-gen")
202          * @return {String} The generated Id.
203          */
204         id : function(el, prefix){
205             prefix = prefix || "roo-gen";
206             el = Roo.getDom(el);
207             var id = prefix + (++idSeed);
208             return el ? (el.id ? el.id : (el.id = id)) : id;
209         },
210          
211        
212         /**
213          * Extends one class with another class and optionally overrides members with the passed literal. This class
214          * also adds the function "override()" to the class that can be used to override
215          * members on an instance.
216          * @param {Object} subclass The class inheriting the functionality
217          * @param {Object} superclass The class being extended
218          * @param {Object} overrides (optional) A literal with members
219          * @method extend
220          */
221         extend : function(){
222             // inline overrides
223             var io = function(o){
224                 for(var m in o){
225                     this[m] = o[m];
226                 }
227             };
228             return function(sb, sp, overrides){
229                 if(typeof sp == 'object'){ // eg. prototype, rather than function constructor..
230                     overrides = sp;
231                     sp = sb;
232                     sb = function(){sp.apply(this, arguments);};
233                 }
234                 var F = function(){}, sbp, spp = sp.prototype;
235                 F.prototype = spp;
236                 sbp = sb.prototype = new F();
237                 sbp.constructor=sb;
238                 sb.superclass=spp;
239                 
240                 if(spp.constructor == Object.prototype.constructor){
241                     spp.constructor=sp;
242                    
243                 }
244                 
245                 sb.override = function(o){
246                     Roo.override(sb, o);
247                 };
248                 sbp.override = io;
249                 Roo.override(sb, overrides);
250                 return sb;
251             };
252         }(),
253
254         /**
255          * Adds a list of functions to the prototype of an existing class, overwriting any existing methods with the same name.
256          * Usage:<pre><code>
257 Roo.override(MyClass, {
258     newMethod1: function(){
259         // etc.
260     },
261     newMethod2: function(foo){
262         // etc.
263     }
264 });
265  </code></pre>
266          * @param {Object} origclass The class to override
267          * @param {Object} overrides The list of functions to add to origClass.  This should be specified as an object literal
268          * containing one or more methods.
269          * @method override
270          */
271         override : function(origclass, overrides){
272             if(overrides){
273                 var p = origclass.prototype;
274                 for(var method in overrides){
275                     p[method] = overrides[method];
276                 }
277             }
278         },
279         /**
280          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
281          * <pre><code>
282 Roo.namespace('Company', 'Company.data');
283 Company.Widget = function() { ... }
284 Company.data.CustomStore = function(config) { ... }
285 </code></pre>
286          * @param {String} namespace1
287          * @param {String} namespace2
288          * @param {String} etc
289          * @method namespace
290          */
291         namespace : function(){
292             var a=arguments, o=null, i, j, d, rt;
293             for (i=0; i<a.length; ++i) {
294                 d=a[i].split(".");
295                 rt = d[0];
296                 /** eval:var:o */
297                 eval('if (typeof ' + rt + ' == "undefined"){' + rt + ' = {};} o = ' + rt + ';');
298                 for (j=1; j<d.length; ++j) {
299                     o[d[j]]=o[d[j]] || {};
300                     o=o[d[j]];
301                 }
302             }
303         },
304         /**
305          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
306          * <pre><code>
307 Roo.factory({ xns: Roo.data, xtype : 'Store', .....});
308 Roo.factory(conf, Roo.data);
309 </code></pre>
310          * @param {String} classname
311          * @param {String} namespace (optional)
312          * @method factory
313          */
314          
315         factory : function(c, ns)
316         {
317             // no xtype, no ns or c.xns - or forced off by c.xns
318             if (!c.xtype   || (!ns && !c.xns) ||  (c.xns === false)) { // not enough info...
319                 return c;
320             }
321             ns = c.xns ? c.xns : ns; // if c.xns is set, then use that..
322             if (c.constructor == ns[c.xtype]) {// already created...
323                 return c;
324             }
325             if (ns[c.xtype]) {
326                 if (Roo.debug) { Roo.log("Roo.Factory(" + c.xtype + ")"); }
327                 var ret = new ns[c.xtype](c);
328                 ret.xns = false;
329                 return ret;
330             }
331             c.xns = false; // prevent recursion..
332             return c;
333         },
334          /**
335          * Logs to console if it can.
336          *
337          * @param {String|Object} string
338          * @method log
339          */
340         log : function(s)
341         {
342             if ((typeof(console) == 'undefined') || (typeof(console.log) == 'undefined')) {
343                 return; // alerT?
344             }
345             
346             console.log(s);
347         },
348         /**
349          * Takes an object and converts it to an encoded URL. e.g. Roo.urlEncode({foo: 1, bar: 2}); would return "foo=1&bar=2".  Optionally, property values can be arrays, instead of keys and the resulting string that's returned will contain a name/value pair for each array value.
350          * @param {Object} o
351          * @return {String}
352          */
353         urlEncode : function(o){
354             if(!o){
355                 return "";
356             }
357             var buf = [];
358             for(var key in o){
359                 var ov = o[key], k = Roo.encodeURIComponent(key);
360                 var type = typeof ov;
361                 if(type == 'undefined'){
362                     buf.push(k, "=&");
363                 }else if(type != "function" && type != "object"){
364                     buf.push(k, "=", Roo.encodeURIComponent(ov), "&");
365                 }else if(ov instanceof Array){
366                     if (ov.length) {
367                             for(var i = 0, len = ov.length; i < len; i++) {
368                                 buf.push(k, "=", Roo.encodeURIComponent(ov[i] === undefined ? '' : ov[i]), "&");
369                             }
370                         } else {
371                             buf.push(k, "=&");
372                         }
373                 }
374             }
375             buf.pop();
376             return buf.join("");
377         },
378          /**
379          * Safe version of encodeURIComponent
380          * @param {String} data 
381          * @return {String} 
382          */
383         
384         encodeURIComponent : function (data)
385         {
386             try {
387                 return encodeURIComponent(data);
388             } catch(e) {} // should be an uri encode error.
389             
390             if (data == '' || data == null){
391                return '';
392             }
393             // http://stackoverflow.com/questions/2596483/unicode-and-uri-encoding-decoding-and-escaping-in-javascript
394             function nibble_to_hex(nibble){
395                 var chars = '0123456789ABCDEF';
396                 return chars.charAt(nibble);
397             }
398             data = data.toString();
399             var buffer = '';
400             for(var i=0; i<data.length; i++){
401                 var c = data.charCodeAt(i);
402                 var bs = new Array();
403                 if (c > 0x10000){
404                         // 4 bytes
405                     bs[0] = 0xF0 | ((c & 0x1C0000) >>> 18);
406                     bs[1] = 0x80 | ((c & 0x3F000) >>> 12);
407                     bs[2] = 0x80 | ((c & 0xFC0) >>> 6);
408                     bs[3] = 0x80 | (c & 0x3F);
409                 }else if (c > 0x800){
410                          // 3 bytes
411                     bs[0] = 0xE0 | ((c & 0xF000) >>> 12);
412                     bs[1] = 0x80 | ((c & 0xFC0) >>> 6);
413                     bs[2] = 0x80 | (c & 0x3F);
414                 }else if (c > 0x80){
415                        // 2 bytes
416                     bs[0] = 0xC0 | ((c & 0x7C0) >>> 6);
417                     bs[1] = 0x80 | (c & 0x3F);
418                 }else{
419                         // 1 byte
420                     bs[0] = c;
421                 }
422                 for(var j=0; j<bs.length; j++){
423                     var b = bs[j];
424                     var hex = nibble_to_hex((b & 0xF0) >>> 4) 
425                             + nibble_to_hex(b &0x0F);
426                     buffer += '%'+hex;
427                }
428             }
429             return buffer;    
430              
431         },
432
433         /**
434          * Takes an encoded URL and and converts it to an object. e.g. Roo.urlDecode("foo=1&bar=2"); would return {foo: 1, bar: 2} or Roo.urlDecode("foo=1&bar=2&bar=3&bar=4", true); would return {foo: 1, bar: [2, 3, 4]}.
435          * @param {String} string
436          * @param {Boolean} overwrite (optional) Items of the same name will overwrite previous values instead of creating an an array (Defaults to false).
437          * @return {Object} A literal with members
438          */
439         urlDecode : function(string, overwrite){
440             if(!string || !string.length){
441                 return {};
442             }
443             var obj = {};
444             var pairs = string.split('&');
445             var pair, name, value;
446             for(var i = 0, len = pairs.length; i < len; i++){
447                 pair = pairs[i].split('=');
448                 name = decodeURIComponent(pair[0]);
449                 value = decodeURIComponent(pair[1]);
450                 if(overwrite !== true){
451                     if(typeof obj[name] == "undefined"){
452                         obj[name] = value;
453                     }else if(typeof obj[name] == "string"){
454                         obj[name] = [obj[name]];
455                         obj[name].push(value);
456                     }else{
457                         obj[name].push(value);
458                     }
459                 }else{
460                     obj[name] = value;
461                 }
462             }
463             return obj;
464         },
465
466         /**
467          * Iterates an array calling the passed function with each item, stopping if your function returns false. If the
468          * passed array is not really an array, your function is called once with it.
469          * The supplied function is called with (Object item, Number index, Array allItems).
470          * @param {Array/NodeList/Mixed} array
471          * @param {Function} fn
472          * @param {Object} scope
473          */
474         each : function(array, fn, scope){
475             if(typeof array.length == "undefined" || typeof array == "string"){
476                 array = [array];
477             }
478             for(var i = 0, len = array.length; i < len; i++){
479                 if(fn.call(scope || array[i], array[i], i, array) === false){ return i; };
480             }
481         },
482
483         // deprecated
484         combine : function(){
485             var as = arguments, l = as.length, r = [];
486             for(var i = 0; i < l; i++){
487                 var a = as[i];
488                 if(a instanceof Array){
489                     r = r.concat(a);
490                 }else if(a.length !== undefined && !a.substr){
491                     r = r.concat(Array.prototype.slice.call(a, 0));
492                 }else{
493                     r.push(a);
494                 }
495             }
496             return r;
497         },
498
499         /**
500          * Escapes the passed string for use in a regular expression
501          * @param {String} str
502          * @return {String}
503          */
504         escapeRe : function(s) {
505             return s.replace(/([.*+?^${}()|[\]\/\\])/g, "\\$1");
506         },
507
508         // internal
509         callback : function(cb, scope, args, delay){
510             if(typeof cb == "function"){
511                 if(delay){
512                     cb.defer(delay, scope, args || []);
513                 }else{
514                     cb.apply(scope, args || []);
515                 }
516             }
517         },
518
519         /**
520          * Return the dom node for the passed string (id), dom node, or Roo.Element
521          * @param {String/HTMLElement/Roo.Element} el
522          * @return HTMLElement
523          */
524         getDom : function(el){
525             if(!el){
526                 return null;
527             }
528             return el.dom ? el.dom : (typeof el == 'string' ? document.getElementById(el) : el);
529         },
530
531         /**
532         * Shorthand for {@link Roo.ComponentMgr#get}
533         * @param {String} id
534         * @return Roo.Component
535         */
536         getCmp : function(id){
537             return Roo.ComponentMgr.get(id);
538         },
539          
540         num : function(v, defaultValue){
541             if(typeof v != 'number'){
542                 return defaultValue;
543             }
544             return v;
545         },
546
547         destroy : function(){
548             for(var i = 0, a = arguments, len = a.length; i < len; i++) {
549                 var as = a[i];
550                 if(as){
551                     if(as.dom){
552                         as.removeAllListeners();
553                         as.remove();
554                         continue;
555                     }
556                     if(typeof as.purgeListeners == 'function'){
557                         as.purgeListeners();
558                     }
559                     if(typeof as.destroy == 'function'){
560                         as.destroy();
561                     }
562                 }
563             }
564         },
565
566         // inpired by a similar function in mootools library
567         /**
568          * Returns the type of object that is passed in. If the object passed in is null or undefined it
569          * return false otherwise it returns one of the following values:<ul>
570          * <li><b>string</b>: If the object passed is a string</li>
571          * <li><b>number</b>: If the object passed is a number</li>
572          * <li><b>boolean</b>: If the object passed is a boolean value</li>
573          * <li><b>function</b>: If the object passed is a function reference</li>
574          * <li><b>object</b>: If the object passed is an object</li>
575          * <li><b>array</b>: If the object passed is an array</li>
576          * <li><b>regexp</b>: If the object passed is a regular expression</li>
577          * <li><b>element</b>: If the object passed is a DOM Element</li>
578          * <li><b>nodelist</b>: If the object passed is a DOM NodeList</li>
579          * <li><b>textnode</b>: If the object passed is a DOM text node and contains something other than whitespace</li>
580          * <li><b>whitespace</b>: If the object passed is a DOM text node and contains only whitespace</li>
581          * @param {Mixed} object
582          * @return {String}
583          */
584         type : function(o){
585             if(o === undefined || o === null){
586                 return false;
587             }
588             if(o.htmlElement){
589                 return 'element';
590             }
591             var t = typeof o;
592             if(t == 'object' && o.nodeName) {
593                 switch(o.nodeType) {
594                     case 1: return 'element';
595                     case 3: return (/\S/).test(o.nodeValue) ? 'textnode' : 'whitespace';
596                 }
597             }
598             if(t == 'object' || t == 'function') {
599                 switch(o.constructor) {
600                     case Array: return 'array';
601                     case RegExp: return 'regexp';
602                 }
603                 if(typeof o.length == 'number' && typeof o.item == 'function') {
604                     return 'nodelist';
605                 }
606             }
607             return t;
608         },
609
610         /**
611          * Returns true if the passed value is null, undefined or an empty string (optional).
612          * @param {Mixed} value The value to test
613          * @param {Boolean} allowBlank (optional) Pass true if an empty string is not considered empty
614          * @return {Boolean}
615          */
616         isEmpty : function(v, allowBlank){
617             return v === null || v === undefined || (!allowBlank ? v === '' : false);
618         },
619         
620         /** @type Boolean */
621         isOpera : isOpera,
622         /** @type Boolean */
623         isSafari : isSafari,
624         /** @type Boolean */
625         isFirefox : isFirefox,
626         /** @type Boolean */
627         isIE : isIE,
628         /** @type Boolean */
629         isIE7 : isIE7,
630         /** @type Boolean */
631         isIE11 : isIE11,
632         /** @type Boolean */
633         isEdge : isEdge,
634         /** @type Boolean */
635         isGecko : isGecko,
636         /** @type Boolean */
637         isBorderBox : isBorderBox,
638         /** @type Boolean */
639         isWindows : isWindows,
640         /** @type Boolean */
641         isLinux : isLinux,
642         /** @type Boolean */
643         isMac : isMac,
644         /** @type Boolean */
645         isIOS : isIOS,
646         /** @type Boolean */
647         isAndroid : isAndroid,
648         /** @type Boolean */
649         isTouch : isTouch,
650
651         /**
652          * By default, Ext intelligently decides whether floating elements should be shimmed. If you are using flash,
653          * you may want to set this to true.
654          * @type Boolean
655          */
656         useShims : ((isIE && !isIE7) || (isGecko && isMac)),
657         
658         
659                 
660         /**
661          * Selects a single element as a Roo Element
662          * This is about as close as you can get to jQuery's $('do crazy stuff')
663          * @param {String} selector The selector/xpath query
664          * @param {Node} root (optional) The start of the query (defaults to document).
665          * @return {Roo.Element}
666          */
667         selectNode : function(selector, root) 
668         {
669             var node = Roo.DomQuery.selectNode(selector,root);
670             return node ? Roo.get(node) : new Roo.Element(false);
671         }
672         
673     });
674
675
676 })();
677
678 Roo.namespace("Roo", "Roo.util", "Roo.grid", "Roo.dd", "Roo.tree", "Roo.data",
679                 "Roo.form", "Roo.menu", "Roo.state", "Roo.lib", "Roo.layout",
680                 "Roo.app", "Roo.ux",
681                 "Roo.bootstrap",
682                 "Roo.bootstrap.dash");
683 /*
684  * Based on:
685  * Ext JS Library 1.1.1
686  * Copyright(c) 2006-2007, Ext JS, LLC.
687  *
688  * Originally Released Under LGPL - original licence link has changed is not relivant.
689  *
690  * Fork - LGPL
691  * <script type="text/javascript">
692  */
693
694 (function() {    
695     // wrappedn so fnCleanup is not in global scope...
696     if(Roo.isIE) {
697         function fnCleanUp() {
698             var p = Function.prototype;
699             delete p.createSequence;
700             delete p.defer;
701             delete p.createDelegate;
702             delete p.createCallback;
703             delete p.createInterceptor;
704
705             window.detachEvent("onunload", fnCleanUp);
706         }
707         window.attachEvent("onunload", fnCleanUp);
708     }
709 })();
710
711
712 /**
713  * @class Function
714  * These functions are available on every Function object (any JavaScript function).
715  */
716 Roo.apply(Function.prototype, {
717      /**
718      * Creates a callback that passes arguments[0], arguments[1], arguments[2], ...
719      * Call directly on any function. Example: <code>myFunction.createCallback(myarg, myarg2)</code>
720      * Will create a function that is bound to those 2 args.
721      * @return {Function} The new function
722     */
723     createCallback : function(/*args...*/){
724         // make args available, in function below
725         var args = arguments;
726         var method = this;
727         return function() {
728             return method.apply(window, args);
729         };
730     },
731
732     /**
733      * Creates a delegate (callback) that sets the scope to obj.
734      * Call directly on any function. Example: <code>this.myFunction.createDelegate(this)</code>
735      * Will create a function that is automatically scoped to this.
736      * @param {Object} obj (optional) The object for which the scope is set
737      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
738      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
739      *                                             if a number the args are inserted at the specified position
740      * @return {Function} The new function
741      */
742     createDelegate : function(obj, args, appendArgs){
743         var method = this;
744         return function() {
745             var callArgs = args || arguments;
746             if(appendArgs === true){
747                 callArgs = Array.prototype.slice.call(arguments, 0);
748                 callArgs = callArgs.concat(args);
749             }else if(typeof appendArgs == "number"){
750                 callArgs = Array.prototype.slice.call(arguments, 0); // copy arguments first
751                 var applyArgs = [appendArgs, 0].concat(args); // create method call params
752                 Array.prototype.splice.apply(callArgs, applyArgs); // splice them in
753             }
754             return method.apply(obj || window, callArgs);
755         };
756     },
757
758     /**
759      * Calls this function after the number of millseconds specified.
760      * @param {Number} millis The number of milliseconds for the setTimeout call (if 0 the function is executed immediately)
761      * @param {Object} obj (optional) The object for which the scope is set
762      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
763      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
764      *                                             if a number the args are inserted at the specified position
765      * @return {Number} The timeout id that can be used with clearTimeout
766      */
767     defer : function(millis, obj, args, appendArgs){
768         var fn = this.createDelegate(obj, args, appendArgs);
769         if(millis){
770             return setTimeout(fn, millis);
771         }
772         fn();
773         return 0;
774     },
775     /**
776      * Create a combined function call sequence of the original function + the passed function.
777      * The resulting function returns the results of the original function.
778      * The passed fcn is called with the parameters of the original function
779      * @param {Function} fcn The function to sequence
780      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
781      * @return {Function} The new function
782      */
783     createSequence : function(fcn, scope){
784         if(typeof fcn != "function"){
785             return this;
786         }
787         var method = this;
788         return function() {
789             var retval = method.apply(this || window, arguments);
790             fcn.apply(scope || this || window, arguments);
791             return retval;
792         };
793     },
794
795     /**
796      * Creates an interceptor function. The passed fcn is called before the original one. If it returns false, the original one is not called.
797      * The resulting function returns the results of the original function.
798      * The passed fcn is called with the parameters of the original function.
799      * @addon
800      * @param {Function} fcn The function to call before the original
801      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
802      * @return {Function} The new function
803      */
804     createInterceptor : function(fcn, scope){
805         if(typeof fcn != "function"){
806             return this;
807         }
808         var method = this;
809         return function() {
810             fcn.target = this;
811             fcn.method = method;
812             if(fcn.apply(scope || this || window, arguments) === false){
813                 return;
814             }
815             return method.apply(this || window, arguments);
816         };
817     }
818 });
819 /*
820  * Based on:
821  * Ext JS Library 1.1.1
822  * Copyright(c) 2006-2007, Ext JS, LLC.
823  *
824  * Originally Released Under LGPL - original licence link has changed is not relivant.
825  *
826  * Fork - LGPL
827  * <script type="text/javascript">
828  */
829
830 Roo.applyIf(String, {
831     
832     /** @scope String */
833     
834     /**
835      * Escapes the passed string for ' and \
836      * @param {String} string The string to escape
837      * @return {String} The escaped string
838      * @static
839      */
840     escape : function(string) {
841         return string.replace(/('|\\)/g, "\\$1");
842     },
843
844     /**
845      * Pads the left side of a string with a specified character.  This is especially useful
846      * for normalizing number and date strings.  Example usage:
847      * <pre><code>
848 var s = String.leftPad('123', 5, '0');
849 // s now contains the string: '00123'
850 </code></pre>
851      * @param {String} string The original string
852      * @param {Number} size The total length of the output string
853      * @param {String} char (optional) The character with which to pad the original string (defaults to empty string " ")
854      * @return {String} The padded string
855      * @static
856      */
857     leftPad : function (val, size, ch) {
858         var result = new String(val);
859         if(ch === null || ch === undefined || ch === '') {
860             ch = " ";
861         }
862         while (result.length < size) {
863             result = ch + result;
864         }
865         return result;
866     },
867
868     /**
869      * Allows you to define a tokenized string and pass an arbitrary number of arguments to replace the tokens.  Each
870      * token must be unique, and must increment in the format {0}, {1}, etc.  Example usage:
871      * <pre><code>
872 var cls = 'my-class', text = 'Some text';
873 var s = String.format('<div class="{0}">{1}</div>', cls, text);
874 // s now contains the string: '<div class="my-class">Some text</div>'
875 </code></pre>
876      * @param {String} string The tokenized string to be formatted
877      * @param {String} value1 The value to replace token {0}
878      * @param {String} value2 Etc...
879      * @return {String} The formatted string
880      * @static
881      */
882     format : function(format){
883         var args = Array.prototype.slice.call(arguments, 1);
884         return format.replace(/\{(\d+)\}/g, function(m, i){
885             return Roo.util.Format.htmlEncode(args[i]);
886         });
887     }
888   
889     
890 });
891
892 /**
893  * Utility function that allows you to easily switch a string between two alternating values.  The passed value
894  * is compared to the current string, and if they are equal, the other value that was passed in is returned.  If
895  * they are already different, the first value passed in is returned.  Note that this method returns the new value
896  * but does not change the current string.
897  * <pre><code>
898 // alternate sort directions
899 sort = sort.toggle('ASC', 'DESC');
900
901 // instead of conditional logic:
902 sort = (sort == 'ASC' ? 'DESC' : 'ASC');
903 </code></pre>
904  * @param {String} value The value to compare to the current string
905  * @param {String} other The new value to use if the string already equals the first value passed in
906  * @return {String} The new value
907  */
908  
909 String.prototype.toggle = function(value, other){
910     return this == value ? other : value;
911 };
912
913
914 /**
915   * Remove invalid unicode characters from a string 
916   *
917   * @return {String} The clean string
918   */
919 String.prototype.unicodeClean = function () {
920     return this.replace(/[\s\S]/g,
921         function(character) {
922             if (character.charCodeAt()< 256) {
923               return character;
924            }
925            try {
926                 encodeURIComponent(character);
927            } catch(e) { 
928               return '';
929            }
930            return character;
931         }
932     );
933 };
934   
935 /*
936  * Based on:
937  * Ext JS Library 1.1.1
938  * Copyright(c) 2006-2007, Ext JS, LLC.
939  *
940  * Originally Released Under LGPL - original licence link has changed is not relivant.
941  *
942  * Fork - LGPL
943  * <script type="text/javascript">
944  */
945
946  /**
947  * @class Number
948  */
949 Roo.applyIf(Number.prototype, {
950     /**
951      * Checks whether or not the current number is within a desired range.  If the number is already within the
952      * range it is returned, otherwise the min or max value is returned depending on which side of the range is
953      * exceeded.  Note that this method returns the constrained value but does not change the current number.
954      * @param {Number} min The minimum number in the range
955      * @param {Number} max The maximum number in the range
956      * @return {Number} The constrained value if outside the range, otherwise the current value
957      */
958     constrain : function(min, max){
959         return Math.min(Math.max(this, min), max);
960     }
961 });/*
962  * Based on:
963  * Ext JS Library 1.1.1
964  * Copyright(c) 2006-2007, Ext JS, LLC.
965  *
966  * Originally Released Under LGPL - original licence link has changed is not relivant.
967  *
968  * Fork - LGPL
969  * <script type="text/javascript">
970  */
971  /**
972  * @class Array
973  */
974 Roo.applyIf(Array.prototype, {
975     /**
976      * 
977      * Checks whether or not the specified object exists in the array.
978      * @param {Object} o The object to check for
979      * @return {Number} The index of o in the array (or -1 if it is not found)
980      */
981     indexOf : function(o){
982        for (var i = 0, len = this.length; i < len; i++){
983               if(this[i] == o) { return i; }
984        }
985            return -1;
986     },
987
988     /**
989      * Removes the specified object from the array.  If the object is not found nothing happens.
990      * @param {Object} o The object to remove
991      */
992     remove : function(o){
993        var index = this.indexOf(o);
994        if(index != -1){
995            this.splice(index, 1);
996        }
997     },
998     /**
999      * Map (JS 1.6 compatibility)
1000      * @param {Function} function  to call
1001      */
1002     map : function(fun )
1003     {
1004         var len = this.length >>> 0;
1005         if (typeof fun != "function") {
1006             throw new TypeError();
1007         }
1008         var res = new Array(len);
1009         var thisp = arguments[1];
1010         for (var i = 0; i < len; i++)
1011         {
1012             if (i in this) {
1013                 res[i] = fun.call(thisp, this[i], i, this);
1014             }
1015         }
1016
1017         return res;
1018     }
1019     
1020 });
1021
1022
1023  
1024 /*
1025  * Based on:
1026  * Ext JS Library 1.1.1
1027  * Copyright(c) 2006-2007, Ext JS, LLC.
1028  *
1029  * Originally Released Under LGPL - original licence link has changed is not relivant.
1030  *
1031  * Fork - LGPL
1032  * <script type="text/javascript">
1033  */
1034
1035 /**
1036  * @class Date
1037  *
1038  * The date parsing and format syntax is a subset of
1039  * <a href="http://www.php.net/date">PHP's date() function</a>, and the formats that are
1040  * supported will provide results equivalent to their PHP versions.
1041  *
1042  * Following is the list of all currently supported formats:
1043  *<pre>
1044 Sample date:
1045 'Wed Jan 10 2007 15:05:01 GMT-0600 (Central Standard Time)'
1046
1047 Format  Output      Description
1048 ------  ----------  --------------------------------------------------------------
1049   d      10         Day of the month, 2 digits with leading zeros
1050   D      Wed        A textual representation of a day, three letters
1051   j      10         Day of the month without leading zeros
1052   l      Wednesday  A full textual representation of the day of the week
1053   S      th         English ordinal day of month suffix, 2 chars (use with j)
1054   w      3          Numeric representation of the day of the week
1055   z      9          The julian date, or day of the year (0-365)
1056   W      01         ISO-8601 2-digit week number of year, weeks starting on Monday (00-52)
1057   F      January    A full textual representation of the month
1058   m      01         Numeric representation of a month, with leading zeros
1059   M      Jan        Month name abbreviation, three letters
1060   n      1          Numeric representation of a month, without leading zeros
1061   t      31         Number of days in the given month
1062   L      0          Whether it's a leap year (1 if it is a leap year, else 0)
1063   Y      2007       A full numeric representation of a year, 4 digits
1064   y      07         A two digit representation of a year
1065   a      pm         Lowercase Ante meridiem and Post meridiem
1066   A      PM         Uppercase Ante meridiem and Post meridiem
1067   g      3          12-hour format of an hour without leading zeros
1068   G      15         24-hour format of an hour without leading zeros
1069   h      03         12-hour format of an hour with leading zeros
1070   H      15         24-hour format of an hour with leading zeros
1071   i      05         Minutes with leading zeros
1072   s      01         Seconds, with leading zeros
1073   O      -0600      Difference to Greenwich time (GMT) in hours (Allows +08, without minutes)
1074   P      -06:00     Difference to Greenwich time (GMT) with colon between hours and minutes
1075   T      CST        Timezone setting of the machine running the code
1076   Z      -21600     Timezone offset in seconds (negative if west of UTC, positive if east)
1077 </pre>
1078  *
1079  * Example usage (note that you must escape format specifiers with '\\' to render them as character literals):
1080  * <pre><code>
1081 var dt = new Date('1/10/2007 03:05:01 PM GMT-0600');
1082 document.write(dt.format('Y-m-d'));                         //2007-01-10
1083 document.write(dt.format('F j, Y, g:i a'));                 //January 10, 2007, 3:05 pm
1084 document.write(dt.format('l, \\t\\he dS of F Y h:i:s A'));  //Wednesday, the 10th of January 2007 03:05:01 PM
1085  </code></pre>
1086  *
1087  * Here are some standard date/time patterns that you might find helpful.  They
1088  * are not part of the source of Date.js, but to use them you can simply copy this
1089  * block of code into any script that is included after Date.js and they will also become
1090  * globally available on the Date object.  Feel free to add or remove patterns as needed in your code.
1091  * <pre><code>
1092 Date.patterns = {
1093     ISO8601Long:"Y-m-d H:i:s",
1094     ISO8601Short:"Y-m-d",
1095     ShortDate: "n/j/Y",
1096     LongDate: "l, F d, Y",
1097     FullDateTime: "l, F d, Y g:i:s A",
1098     MonthDay: "F d",
1099     ShortTime: "g:i A",
1100     LongTime: "g:i:s A",
1101     SortableDateTime: "Y-m-d\\TH:i:s",
1102     UniversalSortableDateTime: "Y-m-d H:i:sO",
1103     YearMonth: "F, Y"
1104 };
1105 </code></pre>
1106  *
1107  * Example usage:
1108  * <pre><code>
1109 var dt = new Date();
1110 document.write(dt.format(Date.patterns.ShortDate));
1111  </code></pre>
1112  */
1113
1114 /*
1115  * Most of the date-formatting functions below are the excellent work of Baron Schwartz.
1116  * They generate precompiled functions from date formats instead of parsing and
1117  * processing the pattern every time you format a date.  These functions are available
1118  * on every Date object (any javascript function).
1119  *
1120  * The original article and download are here:
1121  * http://www.xaprb.com/blog/2005/12/12/javascript-closures-for-runtime-efficiency/
1122  *
1123  */
1124  
1125  
1126  // was in core
1127 /**
1128  Returns the number of milliseconds between this date and date
1129  @param {Date} date (optional) Defaults to now
1130  @return {Number} The diff in milliseconds
1131  @member Date getElapsed
1132  */
1133 Date.prototype.getElapsed = function(date) {
1134         return Math.abs((date || new Date()).getTime()-this.getTime());
1135 };
1136 // was in date file..
1137
1138
1139 // private
1140 Date.parseFunctions = {count:0};
1141 // private
1142 Date.parseRegexes = [];
1143 // private
1144 Date.formatFunctions = {count:0};
1145
1146 // private
1147 Date.prototype.dateFormat = function(format) {
1148     if (Date.formatFunctions[format] == null) {
1149         Date.createNewFormat(format);
1150     }
1151     var func = Date.formatFunctions[format];
1152     return this[func]();
1153 };
1154
1155
1156 /**
1157  * Formats a date given the supplied format string
1158  * @param {String} format The format string
1159  * @return {String} The formatted date
1160  * @method
1161  */
1162 Date.prototype.format = Date.prototype.dateFormat;
1163
1164 // private
1165 Date.createNewFormat = function(format) {
1166     var funcName = "format" + Date.formatFunctions.count++;
1167     Date.formatFunctions[format] = funcName;
1168     var code = "Date.prototype." + funcName + " = function(){return ";
1169     var special = false;
1170     var ch = '';
1171     for (var i = 0; i < format.length; ++i) {
1172         ch = format.charAt(i);
1173         if (!special && ch == "\\") {
1174             special = true;
1175         }
1176         else if (special) {
1177             special = false;
1178             code += "'" + String.escape(ch) + "' + ";
1179         }
1180         else {
1181             code += Date.getFormatCode(ch);
1182         }
1183     }
1184     /** eval:var:zzzzzzzzzzzzz */
1185     eval(code.substring(0, code.length - 3) + ";}");
1186 };
1187
1188 // private
1189 Date.getFormatCode = function(character) {
1190     switch (character) {
1191     case "d":
1192         return "String.leftPad(this.getDate(), 2, '0') + ";
1193     case "D":
1194         return "Date.dayNames[this.getDay()].substring(0, 3) + ";
1195     case "j":
1196         return "this.getDate() + ";
1197     case "l":
1198         return "Date.dayNames[this.getDay()] + ";
1199     case "S":
1200         return "this.getSuffix() + ";
1201     case "w":
1202         return "this.getDay() + ";
1203     case "z":
1204         return "this.getDayOfYear() + ";
1205     case "W":
1206         return "this.getWeekOfYear() + ";
1207     case "F":
1208         return "Date.monthNames[this.getMonth()] + ";
1209     case "m":
1210         return "String.leftPad(this.getMonth() + 1, 2, '0') + ";
1211     case "M":
1212         return "Date.monthNames[this.getMonth()].substring(0, 3) + ";
1213     case "n":
1214         return "(this.getMonth() + 1) + ";
1215     case "t":
1216         return "this.getDaysInMonth() + ";
1217     case "L":
1218         return "(this.isLeapYear() ? 1 : 0) + ";
1219     case "Y":
1220         return "this.getFullYear() + ";
1221     case "y":
1222         return "('' + this.getFullYear()).substring(2, 4) + ";
1223     case "a":
1224         return "(this.getHours() < 12 ? 'am' : 'pm') + ";
1225     case "A":
1226         return "(this.getHours() < 12 ? 'AM' : 'PM') + ";
1227     case "g":
1228         return "((this.getHours() % 12) ? this.getHours() % 12 : 12) + ";
1229     case "G":
1230         return "this.getHours() + ";
1231     case "h":
1232         return "String.leftPad((this.getHours() % 12) ? this.getHours() % 12 : 12, 2, '0') + ";
1233     case "H":
1234         return "String.leftPad(this.getHours(), 2, '0') + ";
1235     case "i":
1236         return "String.leftPad(this.getMinutes(), 2, '0') + ";
1237     case "s":
1238         return "String.leftPad(this.getSeconds(), 2, '0') + ";
1239     case "O":
1240         return "this.getGMTOffset() + ";
1241     case "P":
1242         return "this.getGMTColonOffset() + ";
1243     case "T":
1244         return "this.getTimezone() + ";
1245     case "Z":
1246         return "(this.getTimezoneOffset() * -60) + ";
1247     default:
1248         return "'" + String.escape(character) + "' + ";
1249     }
1250 };
1251
1252 /**
1253  * Parses the passed string using the specified format. Note that this function expects dates in normal calendar
1254  * format, meaning that months are 1-based (1 = January) and not zero-based like in JavaScript dates.  Any part of
1255  * the date format that is not specified will default to the current date value for that part.  Time parts can also
1256  * be specified, but default to 0.  Keep in mind that the input date string must precisely match the specified format
1257  * string or the parse operation will fail.
1258  * Example Usage:
1259 <pre><code>
1260 //dt = Fri May 25 2007 (current date)
1261 var dt = new Date();
1262
1263 //dt = Thu May 25 2006 (today's month/day in 2006)
1264 dt = Date.parseDate("2006", "Y");
1265
1266 //dt = Sun Jan 15 2006 (all date parts specified)
1267 dt = Date.parseDate("2006-1-15", "Y-m-d");
1268
1269 //dt = Sun Jan 15 2006 15:20:01 GMT-0600 (CST)
1270 dt = Date.parseDate("2006-1-15 3:20:01 PM", "Y-m-d h:i:s A" );
1271 </code></pre>
1272  * @param {String} input The unparsed date as a string
1273  * @param {String} format The format the date is in
1274  * @return {Date} The parsed date
1275  * @static
1276  */
1277 Date.parseDate = function(input, format) {
1278     if (Date.parseFunctions[format] == null) {
1279         Date.createParser(format);
1280     }
1281     var func = Date.parseFunctions[format];
1282     return Date[func](input);
1283 };
1284 /**
1285  * @private
1286  */
1287
1288 Date.createParser = function(format) {
1289     var funcName = "parse" + Date.parseFunctions.count++;
1290     var regexNum = Date.parseRegexes.length;
1291     var currentGroup = 1;
1292     Date.parseFunctions[format] = funcName;
1293
1294     var code = "Date." + funcName + " = function(input){\n"
1295         + "var y = -1, m = -1, d = -1, h = -1, i = -1, s = -1, o, z, v;\n"
1296         + "var d = new Date();\n"
1297         + "y = d.getFullYear();\n"
1298         + "m = d.getMonth();\n"
1299         + "d = d.getDate();\n"
1300         + "if (typeof(input) !== 'string') { input = input.toString(); }\n"
1301         + "var results = input.match(Date.parseRegexes[" + regexNum + "]);\n"
1302         + "if (results && results.length > 0) {";
1303     var regex = "";
1304
1305     var special = false;
1306     var ch = '';
1307     for (var i = 0; i < format.length; ++i) {
1308         ch = format.charAt(i);
1309         if (!special && ch == "\\") {
1310             special = true;
1311         }
1312         else if (special) {
1313             special = false;
1314             regex += String.escape(ch);
1315         }
1316         else {
1317             var obj = Date.formatCodeToRegex(ch, currentGroup);
1318             currentGroup += obj.g;
1319             regex += obj.s;
1320             if (obj.g && obj.c) {
1321                 code += obj.c;
1322             }
1323         }
1324     }
1325
1326     code += "if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0 && s >= 0)\n"
1327         + "{v = new Date(y, m, d, h, i, s);}\n"
1328         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0)\n"
1329         + "{v = new Date(y, m, d, h, i);}\n"
1330         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0)\n"
1331         + "{v = new Date(y, m, d, h);}\n"
1332         + "else if (y >= 0 && m >= 0 && d > 0)\n"
1333         + "{v = new Date(y, m, d);}\n"
1334         + "else if (y >= 0 && m >= 0)\n"
1335         + "{v = new Date(y, m);}\n"
1336         + "else if (y >= 0)\n"
1337         + "{v = new Date(y);}\n"
1338         + "}return (v && (z || o))?\n" // favour UTC offset over GMT offset
1339         + "    ((z)? v.add(Date.SECOND, (v.getTimezoneOffset() * 60) + (z*1)) :\n" // reset to UTC, then add offset
1340         + "        v.add(Date.HOUR, (v.getGMTOffset() / 100) + (o / -100))) : v\n" // reset to GMT, then add offset
1341         + ";}";
1342
1343     Date.parseRegexes[regexNum] = new RegExp("^" + regex + "$");
1344     /** eval:var:zzzzzzzzzzzzz */
1345     eval(code);
1346 };
1347
1348 // private
1349 Date.formatCodeToRegex = function(character, currentGroup) {
1350     switch (character) {
1351     case "D":
1352         return {g:0,
1353         c:null,
1354         s:"(?:Sun|Mon|Tue|Wed|Thu|Fri|Sat)"};
1355     case "j":
1356         return {g:1,
1357             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1358             s:"(\\d{1,2})"}; // day of month without leading zeroes
1359     case "d":
1360         return {g:1,
1361             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1362             s:"(\\d{2})"}; // day of month with leading zeroes
1363     case "l":
1364         return {g:0,
1365             c:null,
1366             s:"(?:" + Date.dayNames.join("|") + ")"};
1367     case "S":
1368         return {g:0,
1369             c:null,
1370             s:"(?:st|nd|rd|th)"};
1371     case "w":
1372         return {g:0,
1373             c:null,
1374             s:"\\d"};
1375     case "z":
1376         return {g:0,
1377             c:null,
1378             s:"(?:\\d{1,3})"};
1379     case "W":
1380         return {g:0,
1381             c:null,
1382             s:"(?:\\d{2})"};
1383     case "F":
1384         return {g:1,
1385             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "].substring(0, 3)], 10);\n",
1386             s:"(" + Date.monthNames.join("|") + ")"};
1387     case "M":
1388         return {g:1,
1389             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "]], 10);\n",
1390             s:"(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)"};
1391     case "n":
1392         return {g:1,
1393             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1394             s:"(\\d{1,2})"}; // Numeric representation of a month, without leading zeros
1395     case "m":
1396         return {g:1,
1397             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1398             s:"(\\d{2})"}; // Numeric representation of a month, with leading zeros
1399     case "t":
1400         return {g:0,
1401             c:null,
1402             s:"\\d{1,2}"};
1403     case "L":
1404         return {g:0,
1405             c:null,
1406             s:"(?:1|0)"};
1407     case "Y":
1408         return {g:1,
1409             c:"y = parseInt(results[" + currentGroup + "], 10);\n",
1410             s:"(\\d{4})"};
1411     case "y":
1412         return {g:1,
1413             c:"var ty = parseInt(results[" + currentGroup + "], 10);\n"
1414                 + "y = ty > Date.y2kYear ? 1900 + ty : 2000 + ty;\n",
1415             s:"(\\d{1,2})"};
1416     case "a":
1417         return {g:1,
1418             c:"if (results[" + currentGroup + "] == 'am') {\n"
1419                 + "if (h == 12) { h = 0; }\n"
1420                 + "} else { if (h < 12) { h += 12; }}",
1421             s:"(am|pm)"};
1422     case "A":
1423         return {g:1,
1424             c:"if (results[" + currentGroup + "] == 'AM') {\n"
1425                 + "if (h == 12) { h = 0; }\n"
1426                 + "} else { if (h < 12) { h += 12; }}",
1427             s:"(AM|PM)"};
1428     case "g":
1429     case "G":
1430         return {g:1,
1431             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1432             s:"(\\d{1,2})"}; // 12/24-hr format  format of an hour without leading zeroes
1433     case "h":
1434     case "H":
1435         return {g:1,
1436             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1437             s:"(\\d{2})"}; //  12/24-hr format  format of an hour with leading zeroes
1438     case "i":
1439         return {g:1,
1440             c:"i = parseInt(results[" + currentGroup + "], 10);\n",
1441             s:"(\\d{2})"};
1442     case "s":
1443         return {g:1,
1444             c:"s = parseInt(results[" + currentGroup + "], 10);\n",
1445             s:"(\\d{2})"};
1446     case "O":
1447         return {g:1,
1448             c:[
1449                 "o = results[", currentGroup, "];\n",
1450                 "var sn = o.substring(0,1);\n", // get + / - sign
1451                 "var hr = o.substring(1,3)*1 + Math.floor(o.substring(3,5) / 60);\n", // get hours (performs minutes-to-hour conversion also)
1452                 "var mn = o.substring(3,5) % 60;\n", // get minutes
1453                 "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n", // -12hrs <= GMT offset <= 14hrs
1454                 "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1455             ].join(""),
1456             s:"([+\-]\\d{2,4})"};
1457     
1458     
1459     case "P":
1460         return {g:1,
1461                 c:[
1462                    "o = results[", currentGroup, "];\n",
1463                    "var sn = o.substring(0,1);\n",
1464                    "var hr = o.substring(1,3)*1 + Math.floor(o.substring(4,6) / 60);\n",
1465                    "var mn = o.substring(4,6) % 60;\n",
1466                    "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n",
1467                         "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1468             ].join(""),
1469             s:"([+\-]\\d{4})"};
1470     case "T":
1471         return {g:0,
1472             c:null,
1473             s:"[A-Z]{1,4}"}; // timezone abbrev. may be between 1 - 4 chars
1474     case "Z":
1475         return {g:1,
1476             c:"z = results[" + currentGroup + "];\n" // -43200 <= UTC offset <= 50400
1477                   + "z = (-43200 <= z*1 && z*1 <= 50400)? z : null;\n",
1478             s:"([+\-]?\\d{1,5})"}; // leading '+' sign is optional for UTC offset
1479     default:
1480         return {g:0,
1481             c:null,
1482             s:String.escape(character)};
1483     }
1484 };
1485
1486 /**
1487  * Get the timezone abbreviation of the current date (equivalent to the format specifier 'T').
1488  * @return {String} The abbreviated timezone name (e.g. 'CST')
1489  */
1490 Date.prototype.getTimezone = function() {
1491     return this.toString().replace(/^.*? ([A-Z]{1,4})[\-+][0-9]{4} .*$/, "$1");
1492 };
1493
1494 /**
1495  * Get the offset from GMT of the current date (equivalent to the format specifier 'O').
1496  * @return {String} The 4-character offset string prefixed with + or - (e.g. '-0600')
1497  */
1498 Date.prototype.getGMTOffset = function() {
1499     return (this.getTimezoneOffset() > 0 ? "-" : "+")
1500         + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1501         + String.leftPad(this.getTimezoneOffset() % 60, 2, "0");
1502 };
1503
1504 /**
1505  * Get the offset from GMT of the current date (equivalent to the format specifier 'P').
1506  * @return {String} 2-characters representing hours and 2-characters representing minutes
1507  * seperated by a colon and prefixed with + or - (e.g. '-06:00')
1508  */
1509 Date.prototype.getGMTColonOffset = function() {
1510         return (this.getTimezoneOffset() > 0 ? "-" : "+")
1511                 + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1512                 + ":"
1513                 + String.leftPad(this.getTimezoneOffset() %60, 2, "0");
1514 }
1515
1516 /**
1517  * Get the numeric day number of the year, adjusted for leap year.
1518  * @return {Number} 0 through 364 (365 in leap years)
1519  */
1520 Date.prototype.getDayOfYear = function() {
1521     var num = 0;
1522     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1523     for (var i = 0; i < this.getMonth(); ++i) {
1524         num += Date.daysInMonth[i];
1525     }
1526     return num + this.getDate() - 1;
1527 };
1528
1529 /**
1530  * Get the string representation of the numeric week number of the year
1531  * (equivalent to the format specifier 'W').
1532  * @return {String} '00' through '52'
1533  */
1534 Date.prototype.getWeekOfYear = function() {
1535     // Skip to Thursday of this week
1536     var now = this.getDayOfYear() + (4 - this.getDay());
1537     // Find the first Thursday of the year
1538     var jan1 = new Date(this.getFullYear(), 0, 1);
1539     var then = (7 - jan1.getDay() + 4);
1540     return String.leftPad(((now - then) / 7) + 1, 2, "0");
1541 };
1542
1543 /**
1544  * Whether or not the current date is in a leap year.
1545  * @return {Boolean} True if the current date is in a leap year, else false
1546  */
1547 Date.prototype.isLeapYear = function() {
1548     var year = this.getFullYear();
1549     return ((year & 3) == 0 && (year % 100 || (year % 400 == 0 && year)));
1550 };
1551
1552 /**
1553  * Get the first day of the current month, adjusted for leap year.  The returned value
1554  * is the numeric day index within the week (0-6) which can be used in conjunction with
1555  * the {@link #monthNames} array to retrieve the textual day name.
1556  * Example:
1557  *<pre><code>
1558 var dt = new Date('1/10/2007');
1559 document.write(Date.dayNames[dt.getFirstDayOfMonth()]); //output: 'Monday'
1560 </code></pre>
1561  * @return {Number} The day number (0-6)
1562  */
1563 Date.prototype.getFirstDayOfMonth = function() {
1564     var day = (this.getDay() - (this.getDate() - 1)) % 7;
1565     return (day < 0) ? (day + 7) : day;
1566 };
1567
1568 /**
1569  * Get the last day of the current month, adjusted for leap year.  The returned value
1570  * is the numeric day index within the week (0-6) which can be used in conjunction with
1571  * the {@link #monthNames} array to retrieve the textual day name.
1572  * Example:
1573  *<pre><code>
1574 var dt = new Date('1/10/2007');
1575 document.write(Date.dayNames[dt.getLastDayOfMonth()]); //output: 'Wednesday'
1576 </code></pre>
1577  * @return {Number} The day number (0-6)
1578  */
1579 Date.prototype.getLastDayOfMonth = function() {
1580     var day = (this.getDay() + (Date.daysInMonth[this.getMonth()] - this.getDate())) % 7;
1581     return (day < 0) ? (day + 7) : day;
1582 };
1583
1584
1585 /**
1586  * Get the first date of this date's month
1587  * @return {Date}
1588  */
1589 Date.prototype.getFirstDateOfMonth = function() {
1590     return new Date(this.getFullYear(), this.getMonth(), 1);
1591 };
1592
1593 /**
1594  * Get the last date of this date's month
1595  * @return {Date}
1596  */
1597 Date.prototype.getLastDateOfMonth = function() {
1598     return new Date(this.getFullYear(), this.getMonth(), this.getDaysInMonth());
1599 };
1600 /**
1601  * Get the number of days in the current month, adjusted for leap year.
1602  * @return {Number} The number of days in the month
1603  */
1604 Date.prototype.getDaysInMonth = function() {
1605     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1606     return Date.daysInMonth[this.getMonth()];
1607 };
1608
1609 /**
1610  * Get the English ordinal suffix of the current day (equivalent to the format specifier 'S').
1611  * @return {String} 'st, 'nd', 'rd' or 'th'
1612  */
1613 Date.prototype.getSuffix = function() {
1614     switch (this.getDate()) {
1615         case 1:
1616         case 21:
1617         case 31:
1618             return "st";
1619         case 2:
1620         case 22:
1621             return "nd";
1622         case 3:
1623         case 23:
1624             return "rd";
1625         default:
1626             return "th";
1627     }
1628 };
1629
1630 // private
1631 Date.daysInMonth = [31,28,31,30,31,30,31,31,30,31,30,31];
1632
1633 /**
1634  * An array of textual month names.
1635  * Override these values for international dates, for example...
1636  * Date.monthNames = ['JanInYourLang', 'FebInYourLang', ...];
1637  * @type Array
1638  * @static
1639  */
1640 Date.monthNames =
1641    ["January",
1642     "February",
1643     "March",
1644     "April",
1645     "May",
1646     "June",
1647     "July",
1648     "August",
1649     "September",
1650     "October",
1651     "November",
1652     "December"];
1653
1654 /**
1655  * An array of textual day names.
1656  * Override these values for international dates, for example...
1657  * Date.dayNames = ['SundayInYourLang', 'MondayInYourLang', ...];
1658  * @type Array
1659  * @static
1660  */
1661 Date.dayNames =
1662    ["Sunday",
1663     "Monday",
1664     "Tuesday",
1665     "Wednesday",
1666     "Thursday",
1667     "Friday",
1668     "Saturday"];
1669
1670 // private
1671 Date.y2kYear = 50;
1672 // private
1673 Date.monthNumbers = {
1674     Jan:0,
1675     Feb:1,
1676     Mar:2,
1677     Apr:3,
1678     May:4,
1679     Jun:5,
1680     Jul:6,
1681     Aug:7,
1682     Sep:8,
1683     Oct:9,
1684     Nov:10,
1685     Dec:11};
1686
1687 /**
1688  * Creates and returns a new Date instance with the exact same date value as the called instance.
1689  * Dates are copied and passed by reference, so if a copied date variable is modified later, the original
1690  * variable will also be changed.  When the intention is to create a new variable that will not
1691  * modify the original instance, you should create a clone.
1692  *
1693  * Example of correctly cloning a date:
1694  * <pre><code>
1695 //wrong way:
1696 var orig = new Date('10/1/2006');
1697 var copy = orig;
1698 copy.setDate(5);
1699 document.write(orig);  //returns 'Thu Oct 05 2006'!
1700
1701 //correct way:
1702 var orig = new Date('10/1/2006');
1703 var copy = orig.clone();
1704 copy.setDate(5);
1705 document.write(orig);  //returns 'Thu Oct 01 2006'
1706 </code></pre>
1707  * @return {Date} The new Date instance
1708  */
1709 Date.prototype.clone = function() {
1710         return new Date(this.getTime());
1711 };
1712
1713 /**
1714  * Clears any time information from this date
1715  @param {Boolean} clone true to create a clone of this date, clear the time and return it
1716  @return {Date} this or the clone
1717  */
1718 Date.prototype.clearTime = function(clone){
1719     if(clone){
1720         return this.clone().clearTime();
1721     }
1722     this.setHours(0);
1723     this.setMinutes(0);
1724     this.setSeconds(0);
1725     this.setMilliseconds(0);
1726     return this;
1727 };
1728
1729 // private
1730 // safari setMonth is broken -- check that this is only donw once...
1731 if(Roo.isSafari && typeof(Date.brokenSetMonth) == 'undefined'){
1732     Date.brokenSetMonth = Date.prototype.setMonth;
1733         Date.prototype.setMonth = function(num){
1734                 if(num <= -1){
1735                         var n = Math.ceil(-num);
1736                         var back_year = Math.ceil(n/12);
1737                         var month = (n % 12) ? 12 - n % 12 : 0 ;
1738                         this.setFullYear(this.getFullYear() - back_year);
1739                         return Date.brokenSetMonth.call(this, month);
1740                 } else {
1741                         return Date.brokenSetMonth.apply(this, arguments);
1742                 }
1743         };
1744 }
1745
1746 /** Date interval constant 
1747 * @static 
1748 * @type String */
1749 Date.MILLI = "ms";
1750 /** Date interval constant 
1751 * @static 
1752 * @type String */
1753 Date.SECOND = "s";
1754 /** Date interval constant 
1755 * @static 
1756 * @type String */
1757 Date.MINUTE = "mi";
1758 /** Date interval constant 
1759 * @static 
1760 * @type String */
1761 Date.HOUR = "h";
1762 /** Date interval constant 
1763 * @static 
1764 * @type String */
1765 Date.DAY = "d";
1766 /** Date interval constant 
1767 * @static 
1768 * @type String */
1769 Date.MONTH = "mo";
1770 /** Date interval constant 
1771 * @static 
1772 * @type String */
1773 Date.YEAR = "y";
1774
1775 /**
1776  * Provides a convenient method of performing basic date arithmetic.  This method
1777  * does not modify the Date instance being called - it creates and returns
1778  * a new Date instance containing the resulting date value.
1779  *
1780  * Examples:
1781  * <pre><code>
1782 //Basic usage:
1783 var dt = new Date('10/29/2006').add(Date.DAY, 5);
1784 document.write(dt); //returns 'Fri Oct 06 2006 00:00:00'
1785
1786 //Negative values will subtract correctly:
1787 var dt2 = new Date('10/1/2006').add(Date.DAY, -5);
1788 document.write(dt2); //returns 'Tue Sep 26 2006 00:00:00'
1789
1790 //You can even chain several calls together in one line!
1791 var dt3 = new Date('10/1/2006').add(Date.DAY, 5).add(Date.HOUR, 8).add(Date.MINUTE, -30);
1792 document.write(dt3); //returns 'Fri Oct 06 2006 07:30:00'
1793  </code></pre>
1794  *
1795  * @param {String} interval   A valid date interval enum value
1796  * @param {Number} value      The amount to add to the current date
1797  * @return {Date} The new Date instance
1798  */
1799 Date.prototype.add = function(interval, value){
1800   var d = this.clone();
1801   if (!interval || value === 0) { return d; }
1802   switch(interval.toLowerCase()){
1803     case Date.MILLI:
1804       d.setMilliseconds(this.getMilliseconds() + value);
1805       break;
1806     case Date.SECOND:
1807       d.setSeconds(this.getSeconds() + value);
1808       break;
1809     case Date.MINUTE:
1810       d.setMinutes(this.getMinutes() + value);
1811       break;
1812     case Date.HOUR:
1813       d.setHours(this.getHours() + value);
1814       break;
1815     case Date.DAY:
1816       d.setDate(this.getDate() + value);
1817       break;
1818     case Date.MONTH:
1819       var day = this.getDate();
1820       if(day > 28){
1821           day = Math.min(day, this.getFirstDateOfMonth().add('mo', value).getLastDateOfMonth().getDate());
1822       }
1823       d.setDate(day);
1824       d.setMonth(this.getMonth() + value);
1825       break;
1826     case Date.YEAR:
1827       d.setFullYear(this.getFullYear() + value);
1828       break;
1829   }
1830   return d;
1831 };
1832 /*
1833  * Based on:
1834  * Ext JS Library 1.1.1
1835  * Copyright(c) 2006-2007, Ext JS, LLC.
1836  *
1837  * Originally Released Under LGPL - original licence link has changed is not relivant.
1838  *
1839  * Fork - LGPL
1840  * <script type="text/javascript">
1841  */
1842
1843 /**
1844  * @class Roo.lib.Dom
1845  * @static
1846  * 
1847  * Dom utils (from YIU afaik)
1848  * 
1849  **/
1850 Roo.lib.Dom = {
1851     /**
1852      * Get the view width
1853      * @param {Boolean} full True will get the full document, otherwise it's the view width
1854      * @return {Number} The width
1855      */
1856      
1857     getViewWidth : function(full) {
1858         return full ? this.getDocumentWidth() : this.getViewportWidth();
1859     },
1860     /**
1861      * Get the view height
1862      * @param {Boolean} full True will get the full document, otherwise it's the view height
1863      * @return {Number} The height
1864      */
1865     getViewHeight : function(full) {
1866         return full ? this.getDocumentHeight() : this.getViewportHeight();
1867     },
1868
1869     getDocumentHeight: function() {
1870         var scrollHeight = (document.compatMode != "CSS1Compat") ? document.body.scrollHeight : document.documentElement.scrollHeight;
1871         return Math.max(scrollHeight, this.getViewportHeight());
1872     },
1873
1874     getDocumentWidth: function() {
1875         var scrollWidth = (document.compatMode != "CSS1Compat") ? document.body.scrollWidth : document.documentElement.scrollWidth;
1876         return Math.max(scrollWidth, this.getViewportWidth());
1877     },
1878
1879     getViewportHeight: function() {
1880         var height = self.innerHeight;
1881         var mode = document.compatMode;
1882
1883         if ((mode || Roo.isIE) && !Roo.isOpera) {
1884             height = (mode == "CSS1Compat") ?
1885                      document.documentElement.clientHeight :
1886                      document.body.clientHeight;
1887         }
1888
1889         return height;
1890     },
1891
1892     getViewportWidth: function() {
1893         var width = self.innerWidth;
1894         var mode = document.compatMode;
1895
1896         if (mode || Roo.isIE) {
1897             width = (mode == "CSS1Compat") ?
1898                     document.documentElement.clientWidth :
1899                     document.body.clientWidth;
1900         }
1901         return width;
1902     },
1903
1904     isAncestor : function(p, c) {
1905         p = Roo.getDom(p);
1906         c = Roo.getDom(c);
1907         if (!p || !c) {
1908             return false;
1909         }
1910
1911         if (p.contains && !Roo.isSafari) {
1912             return p.contains(c);
1913         } else if (p.compareDocumentPosition) {
1914             return !!(p.compareDocumentPosition(c) & 16);
1915         } else {
1916             var parent = c.parentNode;
1917             while (parent) {
1918                 if (parent == p) {
1919                     return true;
1920                 }
1921                 else if (!parent.tagName || parent.tagName.toUpperCase() == "HTML") {
1922                     return false;
1923                 }
1924                 parent = parent.parentNode;
1925             }
1926             return false;
1927         }
1928     },
1929
1930     getRegion : function(el) {
1931         return Roo.lib.Region.getRegion(el);
1932     },
1933
1934     getY : function(el) {
1935         return this.getXY(el)[1];
1936     },
1937
1938     getX : function(el) {
1939         return this.getXY(el)[0];
1940     },
1941
1942     getXY : function(el) {
1943         var p, pe, b, scroll, bd = document.body;
1944         el = Roo.getDom(el);
1945         var fly = Roo.lib.AnimBase.fly;
1946         if (el.getBoundingClientRect) {
1947             b = el.getBoundingClientRect();
1948             scroll = fly(document).getScroll();
1949             return [b.left + scroll.left, b.top + scroll.top];
1950         }
1951         var x = 0, y = 0;
1952
1953         p = el;
1954
1955         var hasAbsolute = fly(el).getStyle("position") == "absolute";
1956
1957         while (p) {
1958
1959             x += p.offsetLeft;
1960             y += p.offsetTop;
1961
1962             if (!hasAbsolute && fly(p).getStyle("position") == "absolute") {
1963                 hasAbsolute = true;
1964             }
1965
1966             if (Roo.isGecko) {
1967                 pe = fly(p);
1968
1969                 var bt = parseInt(pe.getStyle("borderTopWidth"), 10) || 0;
1970                 var bl = parseInt(pe.getStyle("borderLeftWidth"), 10) || 0;
1971
1972
1973                 x += bl;
1974                 y += bt;
1975
1976
1977                 if (p != el && pe.getStyle('overflow') != 'visible') {
1978                     x += bl;
1979                     y += bt;
1980                 }
1981             }
1982             p = p.offsetParent;
1983         }
1984
1985         if (Roo.isSafari && hasAbsolute) {
1986             x -= bd.offsetLeft;
1987             y -= bd.offsetTop;
1988         }
1989
1990         if (Roo.isGecko && !hasAbsolute) {
1991             var dbd = fly(bd);
1992             x += parseInt(dbd.getStyle("borderLeftWidth"), 10) || 0;
1993             y += parseInt(dbd.getStyle("borderTopWidth"), 10) || 0;
1994         }
1995
1996         p = el.parentNode;
1997         while (p && p != bd) {
1998             if (!Roo.isOpera || (p.tagName != 'TR' && fly(p).getStyle("display") != "inline")) {
1999                 x -= p.scrollLeft;
2000                 y -= p.scrollTop;
2001             }
2002             p = p.parentNode;
2003         }
2004         return [x, y];
2005     },
2006  
2007   
2008
2009
2010     setXY : function(el, xy) {
2011         el = Roo.fly(el, '_setXY');
2012         el.position();
2013         var pts = el.translatePoints(xy);
2014         if (xy[0] !== false) {
2015             el.dom.style.left = pts.left + "px";
2016         }
2017         if (xy[1] !== false) {
2018             el.dom.style.top = pts.top + "px";
2019         }
2020     },
2021
2022     setX : function(el, x) {
2023         this.setXY(el, [x, false]);
2024     },
2025
2026     setY : function(el, y) {
2027         this.setXY(el, [false, y]);
2028     }
2029 };
2030 /*
2031  * Portions of this file are based on pieces of Yahoo User Interface Library
2032  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2033  * YUI licensed under the BSD License:
2034  * http://developer.yahoo.net/yui/license.txt
2035  * <script type="text/javascript">
2036  *
2037  */
2038
2039 Roo.lib.Event = function() {
2040     var loadComplete = false;
2041     var listeners = [];
2042     var unloadListeners = [];
2043     var retryCount = 0;
2044     var onAvailStack = [];
2045     var counter = 0;
2046     var lastError = null;
2047
2048     return {
2049         POLL_RETRYS: 200,
2050         POLL_INTERVAL: 20,
2051         EL: 0,
2052         TYPE: 1,
2053         FN: 2,
2054         WFN: 3,
2055         OBJ: 3,
2056         ADJ_SCOPE: 4,
2057         _interval: null,
2058
2059         startInterval: function() {
2060             if (!this._interval) {
2061                 var self = this;
2062                 var callback = function() {
2063                     self._tryPreloadAttach();
2064                 };
2065                 this._interval = setInterval(callback, this.POLL_INTERVAL);
2066
2067             }
2068         },
2069
2070         onAvailable: function(p_id, p_fn, p_obj, p_override) {
2071             onAvailStack.push({ id:         p_id,
2072                 fn:         p_fn,
2073                 obj:        p_obj,
2074                 override:   p_override,
2075                 checkReady: false    });
2076
2077             retryCount = this.POLL_RETRYS;
2078             this.startInterval();
2079         },
2080
2081
2082         addListener: function(el, eventName, fn) {
2083             el = Roo.getDom(el);
2084             if (!el || !fn) {
2085                 return false;
2086             }
2087
2088             if ("unload" == eventName) {
2089                 unloadListeners[unloadListeners.length] =
2090                 [el, eventName, fn];
2091                 return true;
2092             }
2093
2094             var wrappedFn = function(e) {
2095                 return fn(Roo.lib.Event.getEvent(e));
2096             };
2097
2098             var li = [el, eventName, fn, wrappedFn];
2099
2100             var index = listeners.length;
2101             listeners[index] = li;
2102
2103             this.doAdd(el, eventName, wrappedFn, false);
2104             return true;
2105
2106         },
2107
2108
2109         removeListener: function(el, eventName, fn) {
2110             var i, len;
2111
2112             el = Roo.getDom(el);
2113
2114             if(!fn) {
2115                 return this.purgeElement(el, false, eventName);
2116             }
2117
2118
2119             if ("unload" == eventName) {
2120
2121                 for (i = 0,len = unloadListeners.length; i < len; i++) {
2122                     var li = unloadListeners[i];
2123                     if (li &&
2124                         li[0] == el &&
2125                         li[1] == eventName &&
2126                         li[2] == fn) {
2127                         unloadListeners.splice(i, 1);
2128                         return true;
2129                     }
2130                 }
2131
2132                 return false;
2133             }
2134
2135             var cacheItem = null;
2136
2137
2138             var index = arguments[3];
2139
2140             if ("undefined" == typeof index) {
2141                 index = this._getCacheIndex(el, eventName, fn);
2142             }
2143
2144             if (index >= 0) {
2145                 cacheItem = listeners[index];
2146             }
2147
2148             if (!el || !cacheItem) {
2149                 return false;
2150             }
2151
2152             this.doRemove(el, eventName, cacheItem[this.WFN], false);
2153
2154             delete listeners[index][this.WFN];
2155             delete listeners[index][this.FN];
2156             listeners.splice(index, 1);
2157
2158             return true;
2159
2160         },
2161
2162
2163         getTarget: function(ev, resolveTextNode) {
2164             ev = ev.browserEvent || ev;
2165             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2166             var t = ev.target || ev.srcElement;
2167             return this.resolveTextNode(t);
2168         },
2169
2170
2171         resolveTextNode: function(node) {
2172             if (Roo.isSafari && node && 3 == node.nodeType) {
2173                 return node.parentNode;
2174             } else {
2175                 return node;
2176             }
2177         },
2178
2179
2180         getPageX: function(ev) {
2181             ev = ev.browserEvent || ev;
2182             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2183             var x = ev.pageX;
2184             if (!x && 0 !== x) {
2185                 x = ev.clientX || 0;
2186
2187                 if (Roo.isIE) {
2188                     x += this.getScroll()[1];
2189                 }
2190             }
2191
2192             return x;
2193         },
2194
2195
2196         getPageY: function(ev) {
2197             ev = ev.browserEvent || ev;
2198             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2199             var y = ev.pageY;
2200             if (!y && 0 !== y) {
2201                 y = ev.clientY || 0;
2202
2203                 if (Roo.isIE) {
2204                     y += this.getScroll()[0];
2205                 }
2206             }
2207
2208
2209             return y;
2210         },
2211
2212
2213         getXY: function(ev) {
2214             ev = ev.browserEvent || ev;
2215             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2216             return [this.getPageX(ev), this.getPageY(ev)];
2217         },
2218
2219
2220         getRelatedTarget: function(ev) {
2221             ev = ev.browserEvent || ev;
2222             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2223             var t = ev.relatedTarget;
2224             if (!t) {
2225                 if (ev.type == "mouseout") {
2226                     t = ev.toElement;
2227                 } else if (ev.type == "mouseover") {
2228                     t = ev.fromElement;
2229                 }
2230             }
2231
2232             return this.resolveTextNode(t);
2233         },
2234
2235
2236         getTime: function(ev) {
2237             ev = ev.browserEvent || ev;
2238             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2239             if (!ev.time) {
2240                 var t = new Date().getTime();
2241                 try {
2242                     ev.time = t;
2243                 } catch(ex) {
2244                     this.lastError = ex;
2245                     return t;
2246                 }
2247             }
2248
2249             return ev.time;
2250         },
2251
2252
2253         stopEvent: function(ev) {
2254             this.stopPropagation(ev);
2255             this.preventDefault(ev);
2256         },
2257
2258
2259         stopPropagation: function(ev) {
2260             ev = ev.browserEvent || ev;
2261             if (ev.stopPropagation) {
2262                 ev.stopPropagation();
2263             } else {
2264                 ev.cancelBubble = true;
2265             }
2266         },
2267
2268
2269         preventDefault: function(ev) {
2270             ev = ev.browserEvent || ev;
2271             if(ev.preventDefault) {
2272                 ev.preventDefault();
2273             } else {
2274                 ev.returnValue = false;
2275             }
2276         },
2277
2278
2279         getEvent: function(e) {
2280             var ev = e || window.event;
2281             if (!ev) {
2282                 var c = this.getEvent.caller;
2283                 while (c) {
2284                     ev = c.arguments[0];
2285                     if (ev && Event == ev.constructor) {
2286                         break;
2287                     }
2288                     c = c.caller;
2289                 }
2290             }
2291             return ev;
2292         },
2293
2294
2295         getCharCode: function(ev) {
2296             ev = ev.browserEvent || ev;
2297             return ev.charCode || ev.keyCode || 0;
2298         },
2299
2300
2301         _getCacheIndex: function(el, eventName, fn) {
2302             for (var i = 0,len = listeners.length; i < len; ++i) {
2303                 var li = listeners[i];
2304                 if (li &&
2305                     li[this.FN] == fn &&
2306                     li[this.EL] == el &&
2307                     li[this.TYPE] == eventName) {
2308                     return i;
2309                 }
2310             }
2311
2312             return -1;
2313         },
2314
2315
2316         elCache: {},
2317
2318
2319         getEl: function(id) {
2320             return document.getElementById(id);
2321         },
2322
2323
2324         clearCache: function() {
2325         },
2326
2327
2328         _load: function(e) {
2329             loadComplete = true;
2330             var EU = Roo.lib.Event;
2331
2332
2333             if (Roo.isIE) {
2334                 EU.doRemove(window, "load", EU._load);
2335             }
2336         },
2337
2338
2339         _tryPreloadAttach: function() {
2340
2341             if (this.locked) {
2342                 return false;
2343             }
2344
2345             this.locked = true;
2346
2347
2348             var tryAgain = !loadComplete;
2349             if (!tryAgain) {
2350                 tryAgain = (retryCount > 0);
2351             }
2352
2353
2354             var notAvail = [];
2355             for (var i = 0,len = onAvailStack.length; i < len; ++i) {
2356                 var item = onAvailStack[i];
2357                 if (item) {
2358                     var el = this.getEl(item.id);
2359
2360                     if (el) {
2361                         if (!item.checkReady ||
2362                             loadComplete ||
2363                             el.nextSibling ||
2364                             (document && document.body)) {
2365
2366                             var scope = el;
2367                             if (item.override) {
2368                                 if (item.override === true) {
2369                                     scope = item.obj;
2370                                 } else {
2371                                     scope = item.override;
2372                                 }
2373                             }
2374                             item.fn.call(scope, item.obj);
2375                             onAvailStack[i] = null;
2376                         }
2377                     } else {
2378                         notAvail.push(item);
2379                     }
2380                 }
2381             }
2382
2383             retryCount = (notAvail.length === 0) ? 0 : retryCount - 1;
2384
2385             if (tryAgain) {
2386
2387                 this.startInterval();
2388             } else {
2389                 clearInterval(this._interval);
2390                 this._interval = null;
2391             }
2392
2393             this.locked = false;
2394
2395             return true;
2396
2397         },
2398
2399
2400         purgeElement: function(el, recurse, eventName) {
2401             var elListeners = this.getListeners(el, eventName);
2402             if (elListeners) {
2403                 for (var i = 0,len = elListeners.length; i < len; ++i) {
2404                     var l = elListeners[i];
2405                     this.removeListener(el, l.type, l.fn);
2406                 }
2407             }
2408
2409             if (recurse && el && el.childNodes) {
2410                 for (i = 0,len = el.childNodes.length; i < len; ++i) {
2411                     this.purgeElement(el.childNodes[i], recurse, eventName);
2412                 }
2413             }
2414         },
2415
2416
2417         getListeners: function(el, eventName) {
2418             var results = [], searchLists;
2419             if (!eventName) {
2420                 searchLists = [listeners, unloadListeners];
2421             } else if (eventName == "unload") {
2422                 searchLists = [unloadListeners];
2423             } else {
2424                 searchLists = [listeners];
2425             }
2426
2427             for (var j = 0; j < searchLists.length; ++j) {
2428                 var searchList = searchLists[j];
2429                 if (searchList && searchList.length > 0) {
2430                     for (var i = 0,len = searchList.length; i < len; ++i) {
2431                         var l = searchList[i];
2432                         if (l && l[this.EL] === el &&
2433                             (!eventName || eventName === l[this.TYPE])) {
2434                             results.push({
2435                                 type:   l[this.TYPE],
2436                                 fn:     l[this.FN],
2437                                 obj:    l[this.OBJ],
2438                                 adjust: l[this.ADJ_SCOPE],
2439                                 index:  i
2440                             });
2441                         }
2442                     }
2443                 }
2444             }
2445
2446             return (results.length) ? results : null;
2447         },
2448
2449
2450         _unload: function(e) {
2451
2452             var EU = Roo.lib.Event, i, j, l, len, index;
2453
2454             for (i = 0,len = unloadListeners.length; i < len; ++i) {
2455                 l = unloadListeners[i];
2456                 if (l) {
2457                     var scope = window;
2458                     if (l[EU.ADJ_SCOPE]) {
2459                         if (l[EU.ADJ_SCOPE] === true) {
2460                             scope = l[EU.OBJ];
2461                         } else {
2462                             scope = l[EU.ADJ_SCOPE];
2463                         }
2464                     }
2465                     l[EU.FN].call(scope, EU.getEvent(e), l[EU.OBJ]);
2466                     unloadListeners[i] = null;
2467                     l = null;
2468                     scope = null;
2469                 }
2470             }
2471
2472             unloadListeners = null;
2473
2474             if (listeners && listeners.length > 0) {
2475                 j = listeners.length;
2476                 while (j) {
2477                     index = j - 1;
2478                     l = listeners[index];
2479                     if (l) {
2480                         EU.removeListener(l[EU.EL], l[EU.TYPE],
2481                                 l[EU.FN], index);
2482                     }
2483                     j = j - 1;
2484                 }
2485                 l = null;
2486
2487                 EU.clearCache();
2488             }
2489
2490             EU.doRemove(window, "unload", EU._unload);
2491
2492         },
2493
2494
2495         getScroll: function() {
2496             var dd = document.documentElement, db = document.body;
2497             if (dd && (dd.scrollTop || dd.scrollLeft)) {
2498                 return [dd.scrollTop, dd.scrollLeft];
2499             } else if (db) {
2500                 return [db.scrollTop, db.scrollLeft];
2501             } else {
2502                 return [0, 0];
2503             }
2504         },
2505
2506
2507         doAdd: function () {
2508             if (window.addEventListener) {
2509                 return function(el, eventName, fn, capture) {
2510                     el.addEventListener(eventName, fn, (capture));
2511                 };
2512             } else if (window.attachEvent) {
2513                 return function(el, eventName, fn, capture) {
2514                     el.attachEvent("on" + eventName, fn);
2515                 };
2516             } else {
2517                 return function() {
2518                 };
2519             }
2520         }(),
2521
2522
2523         doRemove: function() {
2524             if (window.removeEventListener) {
2525                 return function (el, eventName, fn, capture) {
2526                     el.removeEventListener(eventName, fn, (capture));
2527                 };
2528             } else if (window.detachEvent) {
2529                 return function (el, eventName, fn) {
2530                     el.detachEvent("on" + eventName, fn);
2531                 };
2532             } else {
2533                 return function() {
2534                 };
2535             }
2536         }()
2537     };
2538     
2539 }();
2540 (function() {     
2541    
2542     var E = Roo.lib.Event;
2543     E.on = E.addListener;
2544     E.un = E.removeListener;
2545
2546     if (document && document.body) {
2547         E._load();
2548     } else {
2549         E.doAdd(window, "load", E._load);
2550     }
2551     E.doAdd(window, "unload", E._unload);
2552     E._tryPreloadAttach();
2553 })();
2554
2555 /*
2556  * Portions of this file are based on pieces of Yahoo User Interface Library
2557  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2558  * YUI licensed under the BSD License:
2559  * http://developer.yahoo.net/yui/license.txt
2560  * <script type="text/javascript">
2561  *
2562  */
2563
2564 (function() {
2565     /**
2566      * @class Roo.lib.Ajax
2567      *
2568      */
2569     Roo.lib.Ajax = {
2570         /**
2571          * @static 
2572          */
2573         request : function(method, uri, cb, data, options) {
2574             if(options){
2575                 var hs = options.headers;
2576                 if(hs){
2577                     for(var h in hs){
2578                         if(hs.hasOwnProperty(h)){
2579                             this.initHeader(h, hs[h], false);
2580                         }
2581                     }
2582                 }
2583                 if(options.xmlData){
2584                     this.initHeader('Content-Type', 'text/xml', false);
2585                     method = 'POST';
2586                     data = options.xmlData;
2587                 }
2588             }
2589
2590             return this.asyncRequest(method, uri, cb, data);
2591         },
2592
2593         serializeForm : function(form) {
2594             if(typeof form == 'string') {
2595                 form = (document.getElementById(form) || document.forms[form]);
2596             }
2597
2598             var el, name, val, disabled, data = '', hasSubmit = false;
2599             for (var i = 0; i < form.elements.length; i++) {
2600                 el = form.elements[i];
2601                 disabled = form.elements[i].disabled;
2602                 name = form.elements[i].name;
2603                 val = form.elements[i].value;
2604
2605                 if (!disabled && name){
2606                     switch (el.type)
2607                             {
2608                         case 'select-one':
2609                         case 'select-multiple':
2610                             for (var j = 0; j < el.options.length; j++) {
2611                                 if (el.options[j].selected) {
2612                                     if (Roo.isIE) {
2613                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].attributes['value'].specified ? el.options[j].value : el.options[j].text) + '&';
2614                                     }
2615                                     else {
2616                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].hasAttribute('value') ? el.options[j].value : el.options[j].text) + '&';
2617                                     }
2618                                 }
2619                             }
2620                             break;
2621                         case 'radio':
2622                         case 'checkbox':
2623                             if (el.checked) {
2624                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2625                             }
2626                             break;
2627                         case 'file':
2628
2629                         case undefined:
2630
2631                         case 'reset':
2632
2633                         case 'button':
2634
2635                             break;
2636                         case 'submit':
2637                             if(hasSubmit == false) {
2638                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2639                                 hasSubmit = true;
2640                             }
2641                             break;
2642                         default:
2643                             data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2644                             break;
2645                     }
2646                 }
2647             }
2648             data = data.substr(0, data.length - 1);
2649             return data;
2650         },
2651
2652         headers:{},
2653
2654         hasHeaders:false,
2655
2656         useDefaultHeader:true,
2657
2658         defaultPostHeader:'application/x-www-form-urlencoded',
2659
2660         useDefaultXhrHeader:true,
2661
2662         defaultXhrHeader:'XMLHttpRequest',
2663
2664         hasDefaultHeaders:true,
2665
2666         defaultHeaders:{},
2667
2668         poll:{},
2669
2670         timeout:{},
2671
2672         pollInterval:50,
2673
2674         transactionId:0,
2675
2676         setProgId:function(id)
2677         {
2678             this.activeX.unshift(id);
2679         },
2680
2681         setDefaultPostHeader:function(b)
2682         {
2683             this.useDefaultHeader = b;
2684         },
2685
2686         setDefaultXhrHeader:function(b)
2687         {
2688             this.useDefaultXhrHeader = b;
2689         },
2690
2691         setPollingInterval:function(i)
2692         {
2693             if (typeof i == 'number' && isFinite(i)) {
2694                 this.pollInterval = i;
2695             }
2696         },
2697
2698         createXhrObject:function(transactionId)
2699         {
2700             var obj,http;
2701             try
2702             {
2703
2704                 http = new XMLHttpRequest();
2705
2706                 obj = { conn:http, tId:transactionId };
2707             }
2708             catch(e)
2709             {
2710                 for (var i = 0; i < this.activeX.length; ++i) {
2711                     try
2712                     {
2713
2714                         http = new ActiveXObject(this.activeX[i]);
2715
2716                         obj = { conn:http, tId:transactionId };
2717                         break;
2718                     }
2719                     catch(e) {
2720                     }
2721                 }
2722             }
2723             finally
2724             {
2725                 return obj;
2726             }
2727         },
2728
2729         getConnectionObject:function()
2730         {
2731             var o;
2732             var tId = this.transactionId;
2733
2734             try
2735             {
2736                 o = this.createXhrObject(tId);
2737                 if (o) {
2738                     this.transactionId++;
2739                 }
2740             }
2741             catch(e) {
2742             }
2743             finally
2744             {
2745                 return o;
2746             }
2747         },
2748
2749         asyncRequest:function(method, uri, callback, postData)
2750         {
2751             var o = this.getConnectionObject();
2752
2753             if (!o) {
2754                 return null;
2755             }
2756             else {
2757                 o.conn.open(method, uri, true);
2758
2759                 if (this.useDefaultXhrHeader) {
2760                     if (!this.defaultHeaders['X-Requested-With']) {
2761                         this.initHeader('X-Requested-With', this.defaultXhrHeader, true);
2762                     }
2763                 }
2764
2765                 if(postData && this.useDefaultHeader){
2766                     this.initHeader('Content-Type', this.defaultPostHeader);
2767                 }
2768
2769                  if (this.hasDefaultHeaders || this.hasHeaders) {
2770                     this.setHeader(o);
2771                 }
2772
2773                 this.handleReadyState(o, callback);
2774                 o.conn.send(postData || null);
2775
2776                 return o;
2777             }
2778         },
2779
2780         handleReadyState:function(o, callback)
2781         {
2782             var oConn = this;
2783
2784             if (callback && callback.timeout) {
2785                 
2786                 this.timeout[o.tId] = window.setTimeout(function() {
2787                     oConn.abort(o, callback, true);
2788                 }, callback.timeout);
2789             }
2790
2791             this.poll[o.tId] = window.setInterval(
2792                     function() {
2793                         if (o.conn && o.conn.readyState == 4) {
2794                             window.clearInterval(oConn.poll[o.tId]);
2795                             delete oConn.poll[o.tId];
2796
2797                             if(callback && callback.timeout) {
2798                                 window.clearTimeout(oConn.timeout[o.tId]);
2799                                 delete oConn.timeout[o.tId];
2800                             }
2801
2802                             oConn.handleTransactionResponse(o, callback);
2803                         }
2804                     }
2805                     , this.pollInterval);
2806         },
2807
2808         handleTransactionResponse:function(o, callback, isAbort)
2809         {
2810
2811             if (!callback) {
2812                 this.releaseObject(o);
2813                 return;
2814             }
2815
2816             var httpStatus, responseObject;
2817
2818             try
2819             {
2820                 if (o.conn.status !== undefined && o.conn.status != 0) {
2821                     httpStatus = o.conn.status;
2822                 }
2823                 else {
2824                     httpStatus = 13030;
2825                 }
2826             }
2827             catch(e) {
2828
2829
2830                 httpStatus = 13030;
2831             }
2832
2833             if (httpStatus >= 200 && httpStatus < 300) {
2834                 responseObject = this.createResponseObject(o, callback.argument);
2835                 if (callback.success) {
2836                     if (!callback.scope) {
2837                         callback.success(responseObject);
2838                     }
2839                     else {
2840
2841
2842                         callback.success.apply(callback.scope, [responseObject]);
2843                     }
2844                 }
2845             }
2846             else {
2847                 switch (httpStatus) {
2848
2849                     case 12002:
2850                     case 12029:
2851                     case 12030:
2852                     case 12031:
2853                     case 12152:
2854                     case 13030:
2855                         responseObject = this.createExceptionObject(o.tId, callback.argument, (isAbort ? isAbort : false));
2856                         if (callback.failure) {
2857                             if (!callback.scope) {
2858                                 callback.failure(responseObject);
2859                             }
2860                             else {
2861                                 callback.failure.apply(callback.scope, [responseObject]);
2862                             }
2863                         }
2864                         break;
2865                     default:
2866                         responseObject = this.createResponseObject(o, callback.argument);
2867                         if (callback.failure) {
2868                             if (!callback.scope) {
2869                                 callback.failure(responseObject);
2870                             }
2871                             else {
2872                                 callback.failure.apply(callback.scope, [responseObject]);
2873                             }
2874                         }
2875                 }
2876             }
2877
2878             this.releaseObject(o);
2879             responseObject = null;
2880         },
2881
2882         createResponseObject:function(o, callbackArg)
2883         {
2884             var obj = {};
2885             var headerObj = {};
2886
2887             try
2888             {
2889                 var headerStr = o.conn.getAllResponseHeaders();
2890                 var header = headerStr.split('\n');
2891                 for (var i = 0; i < header.length; i++) {
2892                     var delimitPos = header[i].indexOf(':');
2893                     if (delimitPos != -1) {
2894                         headerObj[header[i].substring(0, delimitPos)] = header[i].substring(delimitPos + 2);
2895                     }
2896                 }
2897             }
2898             catch(e) {
2899             }
2900
2901             obj.tId = o.tId;
2902             obj.status = o.conn.status;
2903             obj.statusText = o.conn.statusText;
2904             obj.getResponseHeader = headerObj;
2905             obj.getAllResponseHeaders = headerStr;
2906             obj.responseText = o.conn.responseText;
2907             obj.responseXML = o.conn.responseXML;
2908
2909             if (typeof callbackArg !== undefined) {
2910                 obj.argument = callbackArg;
2911             }
2912
2913             return obj;
2914         },
2915
2916         createExceptionObject:function(tId, callbackArg, isAbort)
2917         {
2918             var COMM_CODE = 0;
2919             var COMM_ERROR = 'communication failure';
2920             var ABORT_CODE = -1;
2921             var ABORT_ERROR = 'transaction aborted';
2922
2923             var obj = {};
2924
2925             obj.tId = tId;
2926             if (isAbort) {
2927                 obj.status = ABORT_CODE;
2928                 obj.statusText = ABORT_ERROR;
2929             }
2930             else {
2931                 obj.status = COMM_CODE;
2932                 obj.statusText = COMM_ERROR;
2933             }
2934
2935             if (callbackArg) {
2936                 obj.argument = callbackArg;
2937             }
2938
2939             return obj;
2940         },
2941
2942         initHeader:function(label, value, isDefault)
2943         {
2944             var headerObj = (isDefault) ? this.defaultHeaders : this.headers;
2945
2946             if (headerObj[label] === undefined) {
2947                 headerObj[label] = value;
2948             }
2949             else {
2950
2951
2952                 headerObj[label] = value + "," + headerObj[label];
2953             }
2954
2955             if (isDefault) {
2956                 this.hasDefaultHeaders = true;
2957             }
2958             else {
2959                 this.hasHeaders = true;
2960             }
2961         },
2962
2963
2964         setHeader:function(o)
2965         {
2966             if (this.hasDefaultHeaders) {
2967                 for (var prop in this.defaultHeaders) {
2968                     if (this.defaultHeaders.hasOwnProperty(prop)) {
2969                         o.conn.setRequestHeader(prop, this.defaultHeaders[prop]);
2970                     }
2971                 }
2972             }
2973
2974             if (this.hasHeaders) {
2975                 for (var prop in this.headers) {
2976                     if (this.headers.hasOwnProperty(prop)) {
2977                         o.conn.setRequestHeader(prop, this.headers[prop]);
2978                     }
2979                 }
2980                 this.headers = {};
2981                 this.hasHeaders = false;
2982             }
2983         },
2984
2985         resetDefaultHeaders:function() {
2986             delete this.defaultHeaders;
2987             this.defaultHeaders = {};
2988             this.hasDefaultHeaders = false;
2989         },
2990
2991         abort:function(o, callback, isTimeout)
2992         {
2993             if(this.isCallInProgress(o)) {
2994                 o.conn.abort();
2995                 window.clearInterval(this.poll[o.tId]);
2996                 delete this.poll[o.tId];
2997                 if (isTimeout) {
2998                     delete this.timeout[o.tId];
2999                 }
3000
3001                 this.handleTransactionResponse(o, callback, true);
3002
3003                 return true;
3004             }
3005             else {
3006                 return false;
3007             }
3008         },
3009
3010
3011         isCallInProgress:function(o)
3012         {
3013             if (o && o.conn) {
3014                 return o.conn.readyState != 4 && o.conn.readyState != 0;
3015             }
3016             else {
3017
3018                 return false;
3019             }
3020         },
3021
3022
3023         releaseObject:function(o)
3024         {
3025
3026             o.conn = null;
3027
3028             o = null;
3029         },
3030
3031         activeX:[
3032         'MSXML2.XMLHTTP.3.0',
3033         'MSXML2.XMLHTTP',
3034         'Microsoft.XMLHTTP'
3035         ]
3036
3037
3038     };
3039 })();/*
3040  * Portions of this file are based on pieces of Yahoo User Interface Library
3041  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3042  * YUI licensed under the BSD License:
3043  * http://developer.yahoo.net/yui/license.txt
3044  * <script type="text/javascript">
3045  *
3046  */
3047
3048 Roo.lib.Region = function(t, r, b, l) {
3049     this.top = t;
3050     this[1] = t;
3051     this.right = r;
3052     this.bottom = b;
3053     this.left = l;
3054     this[0] = l;
3055 };
3056
3057
3058 Roo.lib.Region.prototype = {
3059     contains : function(region) {
3060         return ( region.left >= this.left &&
3061                  region.right <= this.right &&
3062                  region.top >= this.top &&
3063                  region.bottom <= this.bottom    );
3064
3065     },
3066
3067     getArea : function() {
3068         return ( (this.bottom - this.top) * (this.right - this.left) );
3069     },
3070
3071     intersect : function(region) {
3072         var t = Math.max(this.top, region.top);
3073         var r = Math.min(this.right, region.right);
3074         var b = Math.min(this.bottom, region.bottom);
3075         var l = Math.max(this.left, region.left);
3076
3077         if (b >= t && r >= l) {
3078             return new Roo.lib.Region(t, r, b, l);
3079         } else {
3080             return null;
3081         }
3082     },
3083     union : function(region) {
3084         var t = Math.min(this.top, region.top);
3085         var r = Math.max(this.right, region.right);
3086         var b = Math.max(this.bottom, region.bottom);
3087         var l = Math.min(this.left, region.left);
3088
3089         return new Roo.lib.Region(t, r, b, l);
3090     },
3091
3092     adjust : function(t, l, b, r) {
3093         this.top += t;
3094         this.left += l;
3095         this.right += r;
3096         this.bottom += b;
3097         return this;
3098     }
3099 };
3100
3101 Roo.lib.Region.getRegion = function(el) {
3102     var p = Roo.lib.Dom.getXY(el);
3103
3104     var t = p[1];
3105     var r = p[0] + el.offsetWidth;
3106     var b = p[1] + el.offsetHeight;
3107     var l = p[0];
3108
3109     return new Roo.lib.Region(t, r, b, l);
3110 };
3111 /*
3112  * Portions of this file are based on pieces of Yahoo User Interface Library
3113  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3114  * YUI licensed under the BSD License:
3115  * http://developer.yahoo.net/yui/license.txt
3116  * <script type="text/javascript">
3117  *
3118  */
3119 //@@dep Roo.lib.Region
3120
3121
3122 Roo.lib.Point = function(x, y) {
3123     if (x instanceof Array) {
3124         y = x[1];
3125         x = x[0];
3126     }
3127     this.x = this.right = this.left = this[0] = x;
3128     this.y = this.top = this.bottom = this[1] = y;
3129 };
3130
3131 Roo.lib.Point.prototype = new Roo.lib.Region();
3132 /*
3133  * Portions of this file are based on pieces of Yahoo User Interface Library
3134  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3135  * YUI licensed under the BSD License:
3136  * http://developer.yahoo.net/yui/license.txt
3137  * <script type="text/javascript">
3138  *
3139  */
3140  
3141 (function() {   
3142
3143     Roo.lib.Anim = {
3144         scroll : function(el, args, duration, easing, cb, scope) {
3145             this.run(el, args, duration, easing, cb, scope, Roo.lib.Scroll);
3146         },
3147
3148         motion : function(el, args, duration, easing, cb, scope) {
3149             this.run(el, args, duration, easing, cb, scope, Roo.lib.Motion);
3150         },
3151
3152         color : function(el, args, duration, easing, cb, scope) {
3153             this.run(el, args, duration, easing, cb, scope, Roo.lib.ColorAnim);
3154         },
3155
3156         run : function(el, args, duration, easing, cb, scope, type) {
3157             type = type || Roo.lib.AnimBase;
3158             if (typeof easing == "string") {
3159                 easing = Roo.lib.Easing[easing];
3160             }
3161             var anim = new type(el, args, duration, easing);
3162             anim.animateX(function() {
3163                 Roo.callback(cb, scope);
3164             });
3165             return anim;
3166         }
3167     };
3168 })();/*
3169  * Portions of this file are based on pieces of Yahoo User Interface Library
3170  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3171  * YUI licensed under the BSD License:
3172  * http://developer.yahoo.net/yui/license.txt
3173  * <script type="text/javascript">
3174  *
3175  */
3176
3177 (function() {    
3178     var libFlyweight;
3179     
3180     function fly(el) {
3181         if (!libFlyweight) {
3182             libFlyweight = new Roo.Element.Flyweight();
3183         }
3184         libFlyweight.dom = el;
3185         return libFlyweight;
3186     }
3187
3188     // since this uses fly! - it cant be in DOM (which does not have fly yet..)
3189     
3190    
3191     
3192     Roo.lib.AnimBase = function(el, attributes, duration, method) {
3193         if (el) {
3194             this.init(el, attributes, duration, method);
3195         }
3196     };
3197
3198     Roo.lib.AnimBase.fly = fly;
3199     
3200     
3201     
3202     Roo.lib.AnimBase.prototype = {
3203
3204         toString: function() {
3205             var el = this.getEl();
3206             var id = el.id || el.tagName;
3207             return ("Anim " + id);
3208         },
3209
3210         patterns: {
3211             noNegatives:        /width|height|opacity|padding/i,
3212             offsetAttribute:  /^((width|height)|(top|left))$/,
3213             defaultUnit:        /width|height|top$|bottom$|left$|right$/i,
3214             offsetUnit:         /\d+(em|%|en|ex|pt|in|cm|mm|pc)$/i
3215         },
3216
3217
3218         doMethod: function(attr, start, end) {
3219             return this.method(this.currentFrame, start, end - start, this.totalFrames);
3220         },
3221
3222
3223         setAttribute: function(attr, val, unit) {
3224             if (this.patterns.noNegatives.test(attr)) {
3225                 val = (val > 0) ? val : 0;
3226             }
3227
3228             Roo.fly(this.getEl(), '_anim').setStyle(attr, val + unit);
3229         },
3230
3231
3232         getAttribute: function(attr) {
3233             var el = this.getEl();
3234             var val = fly(el).getStyle(attr);
3235
3236             if (val !== 'auto' && !this.patterns.offsetUnit.test(val)) {
3237                 return parseFloat(val);
3238             }
3239
3240             var a = this.patterns.offsetAttribute.exec(attr) || [];
3241             var pos = !!( a[3] );
3242             var box = !!( a[2] );
3243
3244
3245             if (box || (fly(el).getStyle('position') == 'absolute' && pos)) {
3246                 val = el['offset' + a[0].charAt(0).toUpperCase() + a[0].substr(1)];
3247             } else {
3248                 val = 0;
3249             }
3250
3251             return val;
3252         },
3253
3254
3255         getDefaultUnit: function(attr) {
3256             if (this.patterns.defaultUnit.test(attr)) {
3257                 return 'px';
3258             }
3259
3260             return '';
3261         },
3262
3263         animateX : function(callback, scope) {
3264             var f = function() {
3265                 this.onComplete.removeListener(f);
3266                 if (typeof callback == "function") {
3267                     callback.call(scope || this, this);
3268                 }
3269             };
3270             this.onComplete.addListener(f, this);
3271             this.animate();
3272         },
3273
3274
3275         setRuntimeAttribute: function(attr) {
3276             var start;
3277             var end;
3278             var attributes = this.attributes;
3279
3280             this.runtimeAttributes[attr] = {};
3281
3282             var isset = function(prop) {
3283                 return (typeof prop !== 'undefined');
3284             };
3285
3286             if (!isset(attributes[attr]['to']) && !isset(attributes[attr]['by'])) {
3287                 return false;
3288             }
3289
3290             start = ( isset(attributes[attr]['from']) ) ? attributes[attr]['from'] : this.getAttribute(attr);
3291
3292
3293             if (isset(attributes[attr]['to'])) {
3294                 end = attributes[attr]['to'];
3295             } else if (isset(attributes[attr]['by'])) {
3296                 if (start.constructor == Array) {
3297                     end = [];
3298                     for (var i = 0, len = start.length; i < len; ++i) {
3299                         end[i] = start[i] + attributes[attr]['by'][i];
3300                     }
3301                 } else {
3302                     end = start + attributes[attr]['by'];
3303                 }
3304             }
3305
3306             this.runtimeAttributes[attr].start = start;
3307             this.runtimeAttributes[attr].end = end;
3308
3309
3310             this.runtimeAttributes[attr].unit = ( isset(attributes[attr].unit) ) ? attributes[attr]['unit'] : this.getDefaultUnit(attr);
3311         },
3312
3313
3314         init: function(el, attributes, duration, method) {
3315
3316             var isAnimated = false;
3317
3318
3319             var startTime = null;
3320
3321
3322             var actualFrames = 0;
3323
3324
3325             el = Roo.getDom(el);
3326
3327
3328             this.attributes = attributes || {};
3329
3330
3331             this.duration = duration || 1;
3332
3333
3334             this.method = method || Roo.lib.Easing.easeNone;
3335
3336
3337             this.useSeconds = true;
3338
3339
3340             this.currentFrame = 0;
3341
3342
3343             this.totalFrames = Roo.lib.AnimMgr.fps;
3344
3345
3346             this.getEl = function() {
3347                 return el;
3348             };
3349
3350
3351             this.isAnimated = function() {
3352                 return isAnimated;
3353             };
3354
3355
3356             this.getStartTime = function() {
3357                 return startTime;
3358             };
3359
3360             this.runtimeAttributes = {};
3361
3362
3363             this.animate = function() {
3364                 if (this.isAnimated()) {
3365                     return false;
3366                 }
3367
3368                 this.currentFrame = 0;
3369
3370                 this.totalFrames = ( this.useSeconds ) ? Math.ceil(Roo.lib.AnimMgr.fps * this.duration) : this.duration;
3371
3372                 Roo.lib.AnimMgr.registerElement(this);
3373             };
3374
3375
3376             this.stop = function(finish) {
3377                 if (finish) {
3378                     this.currentFrame = this.totalFrames;
3379                     this._onTween.fire();
3380                 }
3381                 Roo.lib.AnimMgr.stop(this);
3382             };
3383
3384             var onStart = function() {
3385                 this.onStart.fire();
3386
3387                 this.runtimeAttributes = {};
3388                 for (var attr in this.attributes) {
3389                     this.setRuntimeAttribute(attr);
3390                 }
3391
3392                 isAnimated = true;
3393                 actualFrames = 0;
3394                 startTime = new Date();
3395             };
3396
3397
3398             var onTween = function() {
3399                 var data = {
3400                     duration: new Date() - this.getStartTime(),
3401                     currentFrame: this.currentFrame
3402                 };
3403
3404                 data.toString = function() {
3405                     return (
3406                             'duration: ' + data.duration +
3407                             ', currentFrame: ' + data.currentFrame
3408                             );
3409                 };
3410
3411                 this.onTween.fire(data);
3412
3413                 var runtimeAttributes = this.runtimeAttributes;
3414
3415                 for (var attr in runtimeAttributes) {
3416                     this.setAttribute(attr, this.doMethod(attr, runtimeAttributes[attr].start, runtimeAttributes[attr].end), runtimeAttributes[attr].unit);
3417                 }
3418
3419                 actualFrames += 1;
3420             };
3421
3422             var onComplete = function() {
3423                 var actual_duration = (new Date() - startTime) / 1000 ;
3424
3425                 var data = {
3426                     duration: actual_duration,
3427                     frames: actualFrames,
3428                     fps: actualFrames / actual_duration
3429                 };
3430
3431                 data.toString = function() {
3432                     return (
3433                             'duration: ' + data.duration +
3434                             ', frames: ' + data.frames +
3435                             ', fps: ' + data.fps
3436                             );
3437                 };
3438
3439                 isAnimated = false;
3440                 actualFrames = 0;
3441                 this.onComplete.fire(data);
3442             };
3443
3444
3445             this._onStart = new Roo.util.Event(this);
3446             this.onStart = new Roo.util.Event(this);
3447             this.onTween = new Roo.util.Event(this);
3448             this._onTween = new Roo.util.Event(this);
3449             this.onComplete = new Roo.util.Event(this);
3450             this._onComplete = new Roo.util.Event(this);
3451             this._onStart.addListener(onStart);
3452             this._onTween.addListener(onTween);
3453             this._onComplete.addListener(onComplete);
3454         }
3455     };
3456 })();
3457 /*
3458  * Portions of this file are based on pieces of Yahoo User Interface Library
3459  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3460  * YUI licensed under the BSD License:
3461  * http://developer.yahoo.net/yui/license.txt
3462  * <script type="text/javascript">
3463  *
3464  */
3465
3466 Roo.lib.AnimMgr = new function() {
3467
3468     var thread = null;
3469
3470
3471     var queue = [];
3472
3473
3474     var tweenCount = 0;
3475
3476
3477     this.fps = 1000;
3478
3479
3480     this.delay = 1;
3481
3482
3483     this.registerElement = function(tween) {
3484         queue[queue.length] = tween;
3485         tweenCount += 1;
3486         tween._onStart.fire();
3487         this.start();
3488     };
3489
3490
3491     this.unRegister = function(tween, index) {
3492         tween._onComplete.fire();
3493         index = index || getIndex(tween);
3494         if (index != -1) {
3495             queue.splice(index, 1);
3496         }
3497
3498         tweenCount -= 1;
3499         if (tweenCount <= 0) {
3500             this.stop();
3501         }
3502     };
3503
3504
3505     this.start = function() {
3506         if (thread === null) {
3507             thread = setInterval(this.run, this.delay);
3508         }
3509     };
3510
3511
3512     this.stop = function(tween) {
3513         if (!tween) {
3514             clearInterval(thread);
3515
3516             for (var i = 0, len = queue.length; i < len; ++i) {
3517                 if (queue[0].isAnimated()) {
3518                     this.unRegister(queue[0], 0);
3519                 }
3520             }
3521
3522             queue = [];
3523             thread = null;
3524             tweenCount = 0;
3525         }
3526         else {
3527             this.unRegister(tween);
3528         }
3529     };
3530
3531
3532     this.run = function() {
3533         for (var i = 0, len = queue.length; i < len; ++i) {
3534             var tween = queue[i];
3535             if (!tween || !tween.isAnimated()) {
3536                 continue;
3537             }
3538
3539             if (tween.currentFrame < tween.totalFrames || tween.totalFrames === null)
3540             {
3541                 tween.currentFrame += 1;
3542
3543                 if (tween.useSeconds) {
3544                     correctFrame(tween);
3545                 }
3546                 tween._onTween.fire();
3547             }
3548             else {
3549                 Roo.lib.AnimMgr.stop(tween, i);
3550             }
3551         }
3552     };
3553
3554     var getIndex = function(anim) {
3555         for (var i = 0, len = queue.length; i < len; ++i) {
3556             if (queue[i] == anim) {
3557                 return i;
3558             }
3559         }
3560         return -1;
3561     };
3562
3563
3564     var correctFrame = function(tween) {
3565         var frames = tween.totalFrames;
3566         var frame = tween.currentFrame;
3567         var expected = (tween.currentFrame * tween.duration * 1000 / tween.totalFrames);
3568         var elapsed = (new Date() - tween.getStartTime());
3569         var tweak = 0;
3570
3571         if (elapsed < tween.duration * 1000) {
3572             tweak = Math.round((elapsed / expected - 1) * tween.currentFrame);
3573         } else {
3574             tweak = frames - (frame + 1);
3575         }
3576         if (tweak > 0 && isFinite(tweak)) {
3577             if (tween.currentFrame + tweak >= frames) {
3578                 tweak = frames - (frame + 1);
3579             }
3580
3581             tween.currentFrame += tweak;
3582         }
3583     };
3584 };
3585
3586     /*
3587  * Portions of this file are based on pieces of Yahoo User Interface Library
3588  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3589  * YUI licensed under the BSD License:
3590  * http://developer.yahoo.net/yui/license.txt
3591  * <script type="text/javascript">
3592  *
3593  */
3594 Roo.lib.Bezier = new function() {
3595
3596         this.getPosition = function(points, t) {
3597             var n = points.length;
3598             var tmp = [];
3599
3600             for (var i = 0; i < n; ++i) {
3601                 tmp[i] = [points[i][0], points[i][1]];
3602             }
3603
3604             for (var j = 1; j < n; ++j) {
3605                 for (i = 0; i < n - j; ++i) {
3606                     tmp[i][0] = (1 - t) * tmp[i][0] + t * tmp[parseInt(i + 1, 10)][0];
3607                     tmp[i][1] = (1 - t) * tmp[i][1] + t * tmp[parseInt(i + 1, 10)][1];
3608                 }
3609             }
3610
3611             return [ tmp[0][0], tmp[0][1] ];
3612
3613         };
3614     };/*
3615  * Portions of this file are based on pieces of Yahoo User Interface Library
3616  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3617  * YUI licensed under the BSD License:
3618  * http://developer.yahoo.net/yui/license.txt
3619  * <script type="text/javascript">
3620  *
3621  */
3622 (function() {
3623
3624     Roo.lib.ColorAnim = function(el, attributes, duration, method) {
3625         Roo.lib.ColorAnim.superclass.constructor.call(this, el, attributes, duration, method);
3626     };
3627
3628     Roo.extend(Roo.lib.ColorAnim, Roo.lib.AnimBase);
3629
3630     var fly = Roo.lib.AnimBase.fly;
3631     var Y = Roo.lib;
3632     var superclass = Y.ColorAnim.superclass;
3633     var proto = Y.ColorAnim.prototype;
3634
3635     proto.toString = function() {
3636         var el = this.getEl();
3637         var id = el.id || el.tagName;
3638         return ("ColorAnim " + id);
3639     };
3640
3641     proto.patterns.color = /color$/i;
3642     proto.patterns.rgb = /^rgb\(([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9]+)\)$/i;
3643     proto.patterns.hex = /^#?([0-9A-F]{2})([0-9A-F]{2})([0-9A-F]{2})$/i;
3644     proto.patterns.hex3 = /^#?([0-9A-F]{1})([0-9A-F]{1})([0-9A-F]{1})$/i;
3645     proto.patterns.transparent = /^transparent|rgba\(0, 0, 0, 0\)$/;
3646
3647
3648     proto.parseColor = function(s) {
3649         if (s.length == 3) {
3650             return s;
3651         }
3652
3653         var c = this.patterns.hex.exec(s);
3654         if (c && c.length == 4) {
3655             return [ parseInt(c[1], 16), parseInt(c[2], 16), parseInt(c[3], 16) ];
3656         }
3657
3658         c = this.patterns.rgb.exec(s);
3659         if (c && c.length == 4) {
3660             return [ parseInt(c[1], 10), parseInt(c[2], 10), parseInt(c[3], 10) ];
3661         }
3662
3663         c = this.patterns.hex3.exec(s);
3664         if (c && c.length == 4) {
3665             return [ parseInt(c[1] + c[1], 16), parseInt(c[2] + c[2], 16), parseInt(c[3] + c[3], 16) ];
3666         }
3667
3668         return null;
3669     };
3670     // since this uses fly! - it cant be in ColorAnim (which does not have fly yet..)
3671     proto.getAttribute = function(attr) {
3672         var el = this.getEl();
3673         if (this.patterns.color.test(attr)) {
3674             var val = fly(el).getStyle(attr);
3675
3676             if (this.patterns.transparent.test(val)) {
3677                 var parent = el.parentNode;
3678                 val = fly(parent).getStyle(attr);
3679
3680                 while (parent && this.patterns.transparent.test(val)) {
3681                     parent = parent.parentNode;
3682                     val = fly(parent).getStyle(attr);
3683                     if (parent.tagName.toUpperCase() == 'HTML') {
3684                         val = '#fff';
3685                     }
3686                 }
3687             }
3688         } else {
3689             val = superclass.getAttribute.call(this, attr);
3690         }
3691
3692         return val;
3693     };
3694     proto.getAttribute = function(attr) {
3695         var el = this.getEl();
3696         if (this.patterns.color.test(attr)) {
3697             var val = fly(el).getStyle(attr);
3698
3699             if (this.patterns.transparent.test(val)) {
3700                 var parent = el.parentNode;
3701                 val = fly(parent).getStyle(attr);
3702
3703                 while (parent && this.patterns.transparent.test(val)) {
3704                     parent = parent.parentNode;
3705                     val = fly(parent).getStyle(attr);
3706                     if (parent.tagName.toUpperCase() == 'HTML') {
3707                         val = '#fff';
3708                     }
3709                 }
3710             }
3711         } else {
3712             val = superclass.getAttribute.call(this, attr);
3713         }
3714
3715         return val;
3716     };
3717
3718     proto.doMethod = function(attr, start, end) {
3719         var val;
3720
3721         if (this.patterns.color.test(attr)) {
3722             val = [];
3723             for (var i = 0, len = start.length; i < len; ++i) {
3724                 val[i] = superclass.doMethod.call(this, attr, start[i], end[i]);
3725             }
3726
3727             val = 'rgb(' + Math.floor(val[0]) + ',' + Math.floor(val[1]) + ',' + Math.floor(val[2]) + ')';
3728         }
3729         else {
3730             val = superclass.doMethod.call(this, attr, start, end);
3731         }
3732
3733         return val;
3734     };
3735
3736     proto.setRuntimeAttribute = function(attr) {
3737         superclass.setRuntimeAttribute.call(this, attr);
3738
3739         if (this.patterns.color.test(attr)) {
3740             var attributes = this.attributes;
3741             var start = this.parseColor(this.runtimeAttributes[attr].start);
3742             var end = this.parseColor(this.runtimeAttributes[attr].end);
3743
3744             if (typeof attributes[attr]['to'] === 'undefined' && typeof attributes[attr]['by'] !== 'undefined') {
3745                 end = this.parseColor(attributes[attr].by);
3746
3747                 for (var i = 0, len = start.length; i < len; ++i) {
3748                     end[i] = start[i] + end[i];
3749                 }
3750             }
3751
3752             this.runtimeAttributes[attr].start = start;
3753             this.runtimeAttributes[attr].end = end;
3754         }
3755     };
3756 })();
3757
3758 /*
3759  * Portions of this file are based on pieces of Yahoo User Interface Library
3760  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3761  * YUI licensed under the BSD License:
3762  * http://developer.yahoo.net/yui/license.txt
3763  * <script type="text/javascript">
3764  *
3765  */
3766 Roo.lib.Easing = {
3767
3768
3769     easeNone: function (t, b, c, d) {
3770         return c * t / d + b;
3771     },
3772
3773
3774     easeIn: function (t, b, c, d) {
3775         return c * (t /= d) * t + b;
3776     },
3777
3778
3779     easeOut: function (t, b, c, d) {
3780         return -c * (t /= d) * (t - 2) + b;
3781     },
3782
3783
3784     easeBoth: function (t, b, c, d) {
3785         if ((t /= d / 2) < 1) {
3786             return c / 2 * t * t + b;
3787         }
3788
3789         return -c / 2 * ((--t) * (t - 2) - 1) + b;
3790     },
3791
3792
3793     easeInStrong: function (t, b, c, d) {
3794         return c * (t /= d) * t * t * t + b;
3795     },
3796
3797
3798     easeOutStrong: function (t, b, c, d) {
3799         return -c * ((t = t / d - 1) * t * t * t - 1) + b;
3800     },
3801
3802
3803     easeBothStrong: function (t, b, c, d) {
3804         if ((t /= d / 2) < 1) {
3805             return c / 2 * t * t * t * t + b;
3806         }
3807
3808         return -c / 2 * ((t -= 2) * t * t * t - 2) + b;
3809     },
3810
3811
3812
3813     elasticIn: function (t, b, c, d, a, p) {
3814         if (t == 0) {
3815             return b;
3816         }
3817         if ((t /= d) == 1) {
3818             return b + c;
3819         }
3820         if (!p) {
3821             p = d * .3;
3822         }
3823
3824         if (!a || a < Math.abs(c)) {
3825             a = c;
3826             var s = p / 4;
3827         }
3828         else {
3829             var s = p / (2 * Math.PI) * Math.asin(c / a);
3830         }
3831
3832         return -(a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3833     },
3834
3835
3836     elasticOut: function (t, b, c, d, a, p) {
3837         if (t == 0) {
3838             return b;
3839         }
3840         if ((t /= d) == 1) {
3841             return b + c;
3842         }
3843         if (!p) {
3844             p = d * .3;
3845         }
3846
3847         if (!a || a < Math.abs(c)) {
3848             a = c;
3849             var s = p / 4;
3850         }
3851         else {
3852             var s = p / (2 * Math.PI) * Math.asin(c / a);
3853         }
3854
3855         return a * Math.pow(2, -10 * t) * Math.sin((t * d - s) * (2 * Math.PI) / p) + c + b;
3856     },
3857
3858
3859     elasticBoth: function (t, b, c, d, a, p) {
3860         if (t == 0) {
3861             return b;
3862         }
3863
3864         if ((t /= d / 2) == 2) {
3865             return b + c;
3866         }
3867
3868         if (!p) {
3869             p = d * (.3 * 1.5);
3870         }
3871
3872         if (!a || a < Math.abs(c)) {
3873             a = c;
3874             var s = p / 4;
3875         }
3876         else {
3877             var s = p / (2 * Math.PI) * Math.asin(c / a);
3878         }
3879
3880         if (t < 1) {
3881             return -.5 * (a * Math.pow(2, 10 * (t -= 1)) *
3882                           Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3883         }
3884         return a * Math.pow(2, -10 * (t -= 1)) *
3885                Math.sin((t * d - s) * (2 * Math.PI) / p) * .5 + c + b;
3886     },
3887
3888
3889
3890     backIn: function (t, b, c, d, s) {
3891         if (typeof s == 'undefined') {
3892             s = 1.70158;
3893         }
3894         return c * (t /= d) * t * ((s + 1) * t - s) + b;
3895     },
3896
3897
3898     backOut: function (t, b, c, d, s) {
3899         if (typeof s == 'undefined') {
3900             s = 1.70158;
3901         }
3902         return c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;
3903     },
3904
3905
3906     backBoth: function (t, b, c, d, s) {
3907         if (typeof s == 'undefined') {
3908             s = 1.70158;
3909         }
3910
3911         if ((t /= d / 2 ) < 1) {
3912             return c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b;
3913         }
3914         return c / 2 * ((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
3915     },
3916
3917
3918     bounceIn: function (t, b, c, d) {
3919         return c - Roo.lib.Easing.bounceOut(d - t, 0, c, d) + b;
3920     },
3921
3922
3923     bounceOut: function (t, b, c, d) {
3924         if ((t /= d) < (1 / 2.75)) {
3925             return c * (7.5625 * t * t) + b;
3926         } else if (t < (2 / 2.75)) {
3927             return c * (7.5625 * (t -= (1.5 / 2.75)) * t + .75) + b;
3928         } else if (t < (2.5 / 2.75)) {
3929             return c * (7.5625 * (t -= (2.25 / 2.75)) * t + .9375) + b;
3930         }
3931         return c * (7.5625 * (t -= (2.625 / 2.75)) * t + .984375) + b;
3932     },
3933
3934
3935     bounceBoth: function (t, b, c, d) {
3936         if (t < d / 2) {
3937             return Roo.lib.Easing.bounceIn(t * 2, 0, c, d) * .5 + b;
3938         }
3939         return Roo.lib.Easing.bounceOut(t * 2 - d, 0, c, d) * .5 + c * .5 + b;
3940     }
3941 };/*
3942  * Portions of this file are based on pieces of Yahoo User Interface Library
3943  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3944  * YUI licensed under the BSD License:
3945  * http://developer.yahoo.net/yui/license.txt
3946  * <script type="text/javascript">
3947  *
3948  */
3949     (function() {
3950         Roo.lib.Motion = function(el, attributes, duration, method) {
3951             if (el) {
3952                 Roo.lib.Motion.superclass.constructor.call(this, el, attributes, duration, method);
3953             }
3954         };
3955
3956         Roo.extend(Roo.lib.Motion, Roo.lib.ColorAnim);
3957
3958
3959         var Y = Roo.lib;
3960         var superclass = Y.Motion.superclass;
3961         var proto = Y.Motion.prototype;
3962
3963         proto.toString = function() {
3964             var el = this.getEl();
3965             var id = el.id || el.tagName;
3966             return ("Motion " + id);
3967         };
3968
3969         proto.patterns.points = /^points$/i;
3970
3971         proto.setAttribute = function(attr, val, unit) {
3972             if (this.patterns.points.test(attr)) {
3973                 unit = unit || 'px';
3974                 superclass.setAttribute.call(this, 'left', val[0], unit);
3975                 superclass.setAttribute.call(this, 'top', val[1], unit);
3976             } else {
3977                 superclass.setAttribute.call(this, attr, val, unit);
3978             }
3979         };
3980
3981         proto.getAttribute = function(attr) {
3982             if (this.patterns.points.test(attr)) {
3983                 var val = [
3984                         superclass.getAttribute.call(this, 'left'),
3985                         superclass.getAttribute.call(this, 'top')
3986                         ];
3987             } else {
3988                 val = superclass.getAttribute.call(this, attr);
3989             }
3990
3991             return val;
3992         };
3993
3994         proto.doMethod = function(attr, start, end) {
3995             var val = null;
3996
3997             if (this.patterns.points.test(attr)) {
3998                 var t = this.method(this.currentFrame, 0, 100, this.totalFrames) / 100;
3999                 val = Y.Bezier.getPosition(this.runtimeAttributes[attr], t);
4000             } else {
4001                 val = superclass.doMethod.call(this, attr, start, end);
4002             }
4003             return val;
4004         };
4005
4006         proto.setRuntimeAttribute = function(attr) {
4007             if (this.patterns.points.test(attr)) {
4008                 var el = this.getEl();
4009                 var attributes = this.attributes;
4010                 var start;
4011                 var control = attributes['points']['control'] || [];
4012                 var end;
4013                 var i, len;
4014
4015                 if (control.length > 0 && !(control[0] instanceof Array)) {
4016                     control = [control];
4017                 } else {
4018                     var tmp = [];
4019                     for (i = 0,len = control.length; i < len; ++i) {
4020                         tmp[i] = control[i];
4021                     }
4022                     control = tmp;
4023                 }
4024
4025                 Roo.fly(el).position();
4026
4027                 if (isset(attributes['points']['from'])) {
4028                     Roo.lib.Dom.setXY(el, attributes['points']['from']);
4029                 }
4030                 else {
4031                     Roo.lib.Dom.setXY(el, Roo.lib.Dom.getXY(el));
4032                 }
4033
4034                 start = this.getAttribute('points');
4035
4036
4037                 if (isset(attributes['points']['to'])) {
4038                     end = translateValues.call(this, attributes['points']['to'], start);
4039
4040                     var pageXY = Roo.lib.Dom.getXY(this.getEl());
4041                     for (i = 0,len = control.length; i < len; ++i) {
4042                         control[i] = translateValues.call(this, control[i], start);
4043                     }
4044
4045
4046                 } else if (isset(attributes['points']['by'])) {
4047                     end = [ start[0] + attributes['points']['by'][0], start[1] + attributes['points']['by'][1] ];
4048
4049                     for (i = 0,len = control.length; i < len; ++i) {
4050                         control[i] = [ start[0] + control[i][0], start[1] + control[i][1] ];
4051                     }
4052                 }
4053
4054                 this.runtimeAttributes[attr] = [start];
4055
4056                 if (control.length > 0) {
4057                     this.runtimeAttributes[attr] = this.runtimeAttributes[attr].concat(control);
4058                 }
4059
4060                 this.runtimeAttributes[attr][this.runtimeAttributes[attr].length] = end;
4061             }
4062             else {
4063                 superclass.setRuntimeAttribute.call(this, attr);
4064             }
4065         };
4066
4067         var translateValues = function(val, start) {
4068             var pageXY = Roo.lib.Dom.getXY(this.getEl());
4069             val = [ val[0] - pageXY[0] + start[0], val[1] - pageXY[1] + start[1] ];
4070
4071             return val;
4072         };
4073
4074         var isset = function(prop) {
4075             return (typeof prop !== 'undefined');
4076         };
4077     })();
4078 /*
4079  * Portions of this file are based on pieces of Yahoo User Interface Library
4080  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
4081  * YUI licensed under the BSD License:
4082  * http://developer.yahoo.net/yui/license.txt
4083  * <script type="text/javascript">
4084  *
4085  */
4086     (function() {
4087         Roo.lib.Scroll = function(el, attributes, duration, method) {
4088             if (el) {
4089                 Roo.lib.Scroll.superclass.constructor.call(this, el, attributes, duration, method);
4090             }
4091         };
4092
4093         Roo.extend(Roo.lib.Scroll, Roo.lib.ColorAnim);
4094
4095
4096         var Y = Roo.lib;
4097         var superclass = Y.Scroll.superclass;
4098         var proto = Y.Scroll.prototype;
4099
4100         proto.toString = function() {
4101             var el = this.getEl();
4102             var id = el.id || el.tagName;
4103             return ("Scroll " + id);
4104         };
4105
4106         proto.doMethod = function(attr, start, end) {
4107             var val = null;
4108
4109             if (attr == 'scroll') {
4110                 val = [
4111                         this.method(this.currentFrame, start[0], end[0] - start[0], this.totalFrames),
4112                         this.method(this.currentFrame, start[1], end[1] - start[1], this.totalFrames)
4113                         ];
4114
4115             } else {
4116                 val = superclass.doMethod.call(this, attr, start, end);
4117             }
4118             return val;
4119         };
4120
4121         proto.getAttribute = function(attr) {
4122             var val = null;
4123             var el = this.getEl();
4124
4125             if (attr == 'scroll') {
4126                 val = [ el.scrollLeft, el.scrollTop ];
4127             } else {
4128                 val = superclass.getAttribute.call(this, attr);
4129             }
4130
4131             return val;
4132         };
4133
4134         proto.setAttribute = function(attr, val, unit) {
4135             var el = this.getEl();
4136
4137             if (attr == 'scroll') {
4138                 el.scrollLeft = val[0];
4139                 el.scrollTop = val[1];
4140             } else {
4141                 superclass.setAttribute.call(this, attr, val, unit);
4142             }
4143         };
4144     })();
4145 /*
4146  * Based on:
4147  * Ext JS Library 1.1.1
4148  * Copyright(c) 2006-2007, Ext JS, LLC.
4149  *
4150  * Originally Released Under LGPL - original licence link has changed is not relivant.
4151  *
4152  * Fork - LGPL
4153  * <script type="text/javascript">
4154  */
4155
4156
4157 // nasty IE9 hack - what a pile of crap that is..
4158
4159  if (typeof Range != "undefined" && typeof Range.prototype.createContextualFragment == "undefined") {
4160     Range.prototype.createContextualFragment = function (html) {
4161         var doc = window.document;
4162         var container = doc.createElement("div");
4163         container.innerHTML = html;
4164         var frag = doc.createDocumentFragment(), n;
4165         while ((n = container.firstChild)) {
4166             frag.appendChild(n);
4167         }
4168         return frag;
4169     };
4170 }
4171
4172 /**
4173  * @class Roo.DomHelper
4174  * Utility class for working with DOM and/or Templates. It transparently supports using HTML fragments or DOM.
4175  * For more information see <a href="http://web.archive.org/web/20071221063734/http://www.jackslocum.com/blog/2006/10/06/domhelper-create-elements-using-dom-html-fragments-or-templates/">this blog post with examples</a>.
4176  * @singleton
4177  */
4178 Roo.DomHelper = function(){
4179     var tempTableEl = null;
4180     var emptyTags = /^(?:br|frame|hr|img|input|link|meta|range|spacer|wbr|area|param|col)$/i;
4181     var tableRe = /^table|tbody|tr|td$/i;
4182     var xmlns = {};
4183     // build as innerHTML where available
4184     /** @ignore */
4185     var createHtml = function(o){
4186         if(typeof o == 'string'){
4187             return o;
4188         }
4189         var b = "";
4190         if(!o.tag){
4191             o.tag = "div";
4192         }
4193         b += "<" + o.tag;
4194         for(var attr in o){
4195             if(attr == "tag" || attr == "children" || attr == "cn" || attr == "html" || typeof o[attr] == "function") { continue; }
4196             if(attr == "style"){
4197                 var s = o["style"];
4198                 if(typeof s == "function"){
4199                     s = s.call();
4200                 }
4201                 if(typeof s == "string"){
4202                     b += ' style="' + s + '"';
4203                 }else if(typeof s == "object"){
4204                     b += ' style="';
4205                     for(var key in s){
4206                         if(typeof s[key] != "function"){
4207                             b += key + ":" + s[key] + ";";
4208                         }
4209                     }
4210                     b += '"';
4211                 }
4212             }else{
4213                 if(attr == "cls"){
4214                     b += ' class="' + o["cls"] + '"';
4215                 }else if(attr == "htmlFor"){
4216                     b += ' for="' + o["htmlFor"] + '"';
4217                 }else{
4218                     b += " " + attr + '="' + o[attr] + '"';
4219                 }
4220             }
4221         }
4222         if(emptyTags.test(o.tag)){
4223             b += "/>";
4224         }else{
4225             b += ">";
4226             var cn = o.children || o.cn;
4227             if(cn){
4228                 //http://bugs.kde.org/show_bug.cgi?id=71506
4229                 if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4230                     for(var i = 0, len = cn.length; i < len; i++) {
4231                         b += createHtml(cn[i], b);
4232                     }
4233                 }else{
4234                     b += createHtml(cn, b);
4235                 }
4236             }
4237             if(o.html){
4238                 b += o.html;
4239             }
4240             b += "</" + o.tag + ">";
4241         }
4242         return b;
4243     };
4244
4245     // build as dom
4246     /** @ignore */
4247     var createDom = function(o, parentNode){
4248          
4249         // defininition craeted..
4250         var ns = false;
4251         if (o.ns && o.ns != 'html') {
4252                
4253             if (o.xmlns && typeof(xmlns[o.ns]) == 'undefined') {
4254                 xmlns[o.ns] = o.xmlns;
4255                 ns = o.xmlns;
4256             }
4257             if (typeof(xmlns[o.ns]) == 'undefined') {
4258                 console.log("Trying to create namespace element " + o.ns + ", however no xmlns was sent to builder previously");
4259             }
4260             ns = xmlns[o.ns];
4261         }
4262         
4263         
4264         if (typeof(o) == 'string') {
4265             return parentNode.appendChild(document.createTextNode(o));
4266         }
4267         o.tag = o.tag || div;
4268         if (o.ns && Roo.isIE) {
4269             ns = false;
4270             o.tag = o.ns + ':' + o.tag;
4271             
4272         }
4273         var el = ns ? document.createElementNS( ns, o.tag||'div') :  document.createElement(o.tag||'div');
4274         var useSet = el.setAttribute ? true : false; // In IE some elements don't have setAttribute
4275         for(var attr in o){
4276             
4277             if(attr == "tag" || attr == "ns" ||attr == "xmlns" ||attr == "children" || attr == "cn" || attr == "html" || 
4278                     attr == "style" || typeof o[attr] == "function") { continue; }
4279                     
4280             if(attr=="cls" && Roo.isIE){
4281                 el.className = o["cls"];
4282             }else{
4283                 if(useSet) { el.setAttribute(attr=="cls" ? 'class' : attr, o[attr]);}
4284                 else { 
4285                     el[attr] = o[attr];
4286                 }
4287             }
4288         }
4289         Roo.DomHelper.applyStyles(el, o.style);
4290         var cn = o.children || o.cn;
4291         if(cn){
4292             //http://bugs.kde.org/show_bug.cgi?id=71506
4293              if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4294                 for(var i = 0, len = cn.length; i < len; i++) {
4295                     createDom(cn[i], el);
4296                 }
4297             }else{
4298                 createDom(cn, el);
4299             }
4300         }
4301         if(o.html){
4302             el.innerHTML = o.html;
4303         }
4304         if(parentNode){
4305            parentNode.appendChild(el);
4306         }
4307         return el;
4308     };
4309
4310     var ieTable = function(depth, s, h, e){
4311         tempTableEl.innerHTML = [s, h, e].join('');
4312         var i = -1, el = tempTableEl;
4313         while(++i < depth){
4314             el = el.firstChild;
4315         }
4316         return el;
4317     };
4318
4319     // kill repeat to save bytes
4320     var ts = '<table>',
4321         te = '</table>',
4322         tbs = ts+'<tbody>',
4323         tbe = '</tbody>'+te,
4324         trs = tbs + '<tr>',
4325         tre = '</tr>'+tbe;
4326
4327     /**
4328      * @ignore
4329      * Nasty code for IE's broken table implementation
4330      */
4331     var insertIntoTable = function(tag, where, el, html){
4332         if(!tempTableEl){
4333             tempTableEl = document.createElement('div');
4334         }
4335         var node;
4336         var before = null;
4337         if(tag == 'td'){
4338             if(where == 'afterbegin' || where == 'beforeend'){ // INTO a TD
4339                 return;
4340             }
4341             if(where == 'beforebegin'){
4342                 before = el;
4343                 el = el.parentNode;
4344             } else{
4345                 before = el.nextSibling;
4346                 el = el.parentNode;
4347             }
4348             node = ieTable(4, trs, html, tre);
4349         }
4350         else if(tag == 'tr'){
4351             if(where == 'beforebegin'){
4352                 before = el;
4353                 el = el.parentNode;
4354                 node = ieTable(3, tbs, html, tbe);
4355             } else if(where == 'afterend'){
4356                 before = el.nextSibling;
4357                 el = el.parentNode;
4358                 node = ieTable(3, tbs, html, tbe);
4359             } else{ // INTO a TR
4360                 if(where == 'afterbegin'){
4361                     before = el.firstChild;
4362                 }
4363                 node = ieTable(4, trs, html, tre);
4364             }
4365         } else if(tag == 'tbody'){
4366             if(where == 'beforebegin'){
4367                 before = el;
4368                 el = el.parentNode;
4369                 node = ieTable(2, ts, html, te);
4370             } else if(where == 'afterend'){
4371                 before = el.nextSibling;
4372                 el = el.parentNode;
4373                 node = ieTable(2, ts, html, te);
4374             } else{
4375                 if(where == 'afterbegin'){
4376                     before = el.firstChild;
4377                 }
4378                 node = ieTable(3, tbs, html, tbe);
4379             }
4380         } else{ // TABLE
4381             if(where == 'beforebegin' || where == 'afterend'){ // OUTSIDE the table
4382                 return;
4383             }
4384             if(where == 'afterbegin'){
4385                 before = el.firstChild;
4386             }
4387             node = ieTable(2, ts, html, te);
4388         }
4389         el.insertBefore(node, before);
4390         return node;
4391     };
4392
4393     return {
4394     /** True to force the use of DOM instead of html fragments @type Boolean */
4395     useDom : false,
4396
4397     /**
4398      * Returns the markup for the passed Element(s) config
4399      * @param {Object} o The Dom object spec (and children)
4400      * @return {String}
4401      */
4402     markup : function(o){
4403         return createHtml(o);
4404     },
4405
4406     /**
4407      * Applies a style specification to an element
4408      * @param {String/HTMLElement} el The element to apply styles to
4409      * @param {String/Object/Function} styles A style specification string eg "width:100px", or object in the form {width:"100px"}, or
4410      * a function which returns such a specification.
4411      */
4412     applyStyles : function(el, styles){
4413         if(styles){
4414            el = Roo.fly(el);
4415            if(typeof styles == "string"){
4416                var re = /\s?([a-z\-]*)\:\s?([^;]*);?/gi;
4417                var matches;
4418                while ((matches = re.exec(styles)) != null){
4419                    el.setStyle(matches[1], matches[2]);
4420                }
4421            }else if (typeof styles == "object"){
4422                for (var style in styles){
4423                   el.setStyle(style, styles[style]);
4424                }
4425            }else if (typeof styles == "function"){
4426                 Roo.DomHelper.applyStyles(el, styles.call());
4427            }
4428         }
4429     },
4430
4431     /**
4432      * Inserts an HTML fragment into the Dom
4433      * @param {String} where Where to insert the html in relation to el - beforeBegin, afterBegin, beforeEnd, afterEnd.
4434      * @param {HTMLElement} el The context element
4435      * @param {String} html The HTML fragmenet
4436      * @return {HTMLElement} The new node
4437      */
4438     insertHtml : function(where, el, html){
4439         where = where.toLowerCase();
4440         if(el.insertAdjacentHTML){
4441             if(tableRe.test(el.tagName)){
4442                 var rs;
4443                 if(rs = insertIntoTable(el.tagName.toLowerCase(), where, el, html)){
4444                     return rs;
4445                 }
4446             }
4447             switch(where){
4448                 case "beforebegin":
4449                     el.insertAdjacentHTML('BeforeBegin', html);
4450                     return el.previousSibling;
4451                 case "afterbegin":
4452                     el.insertAdjacentHTML('AfterBegin', html);
4453                     return el.firstChild;
4454                 case "beforeend":
4455                     el.insertAdjacentHTML('BeforeEnd', html);
4456                     return el.lastChild;
4457                 case "afterend":
4458                     el.insertAdjacentHTML('AfterEnd', html);
4459                     return el.nextSibling;
4460             }
4461             throw 'Illegal insertion point -> "' + where + '"';
4462         }
4463         var range = el.ownerDocument.createRange();
4464         var frag;
4465         switch(where){
4466              case "beforebegin":
4467                 range.setStartBefore(el);
4468                 frag = range.createContextualFragment(html);
4469                 el.parentNode.insertBefore(frag, el);
4470                 return el.previousSibling;
4471              case "afterbegin":
4472                 if(el.firstChild){
4473                     range.setStartBefore(el.firstChild);
4474                     frag = range.createContextualFragment(html);
4475                     el.insertBefore(frag, el.firstChild);
4476                     return el.firstChild;
4477                 }else{
4478                     el.innerHTML = html;
4479                     return el.firstChild;
4480                 }
4481             case "beforeend":
4482                 if(el.lastChild){
4483                     range.setStartAfter(el.lastChild);
4484                     frag = range.createContextualFragment(html);
4485                     el.appendChild(frag);
4486                     return el.lastChild;
4487                 }else{
4488                     el.innerHTML = html;
4489                     return el.lastChild;
4490                 }
4491             case "afterend":
4492                 range.setStartAfter(el);
4493                 frag = range.createContextualFragment(html);
4494                 el.parentNode.insertBefore(frag, el.nextSibling);
4495                 return el.nextSibling;
4496             }
4497             throw 'Illegal insertion point -> "' + where + '"';
4498     },
4499
4500     /**
4501      * Creates new Dom element(s) and inserts them before el
4502      * @param {String/HTMLElement/Element} el The context element
4503      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4504      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4505      * @return {HTMLElement/Roo.Element} The new node
4506      */
4507     insertBefore : function(el, o, returnElement){
4508         return this.doInsert(el, o, returnElement, "beforeBegin");
4509     },
4510
4511     /**
4512      * Creates new Dom element(s) and inserts them after el
4513      * @param {String/HTMLElement/Element} el The context element
4514      * @param {Object} o The Dom object spec (and children)
4515      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4516      * @return {HTMLElement/Roo.Element} The new node
4517      */
4518     insertAfter : function(el, o, returnElement){
4519         return this.doInsert(el, o, returnElement, "afterEnd", "nextSibling");
4520     },
4521
4522     /**
4523      * Creates new Dom element(s) and inserts them as the first child of el
4524      * @param {String/HTMLElement/Element} el The context element
4525      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4526      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4527      * @return {HTMLElement/Roo.Element} The new node
4528      */
4529     insertFirst : function(el, o, returnElement){
4530         return this.doInsert(el, o, returnElement, "afterBegin");
4531     },
4532
4533     // private
4534     doInsert : function(el, o, returnElement, pos, sibling){
4535         el = Roo.getDom(el);
4536         var newNode;
4537         if(this.useDom || o.ns){
4538             newNode = createDom(o, null);
4539             el.parentNode.insertBefore(newNode, sibling ? el[sibling] : el);
4540         }else{
4541             var html = createHtml(o);
4542             newNode = this.insertHtml(pos, el, html);
4543         }
4544         return returnElement ? Roo.get(newNode, true) : newNode;
4545     },
4546
4547     /**
4548      * Creates new Dom element(s) and appends them to el
4549      * @param {String/HTMLElement/Element} el The context element
4550      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4551      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4552      * @return {HTMLElement/Roo.Element} The new node
4553      */
4554     append : function(el, o, returnElement){
4555         el = Roo.getDom(el);
4556         var newNode;
4557         if(this.useDom || o.ns){
4558             newNode = createDom(o, null);
4559             el.appendChild(newNode);
4560         }else{
4561             var html = createHtml(o);
4562             newNode = this.insertHtml("beforeEnd", el, html);
4563         }
4564         return returnElement ? Roo.get(newNode, true) : newNode;
4565     },
4566
4567     /**
4568      * Creates new Dom element(s) and overwrites the contents of el with them
4569      * @param {String/HTMLElement/Element} el The context element
4570      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4571      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4572      * @return {HTMLElement/Roo.Element} The new node
4573      */
4574     overwrite : function(el, o, returnElement){
4575         el = Roo.getDom(el);
4576         if (o.ns) {
4577           
4578             while (el.childNodes.length) {
4579                 el.removeChild(el.firstChild);
4580             }
4581             createDom(o, el);
4582         } else {
4583             el.innerHTML = createHtml(o);   
4584         }
4585         
4586         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4587     },
4588
4589     /**
4590      * Creates a new Roo.DomHelper.Template from the Dom object spec
4591      * @param {Object} o The Dom object spec (and children)
4592      * @return {Roo.DomHelper.Template} The new template
4593      */
4594     createTemplate : function(o){
4595         var html = createHtml(o);
4596         return new Roo.Template(html);
4597     }
4598     };
4599 }();
4600 /*
4601  * Based on:
4602  * Ext JS Library 1.1.1
4603  * Copyright(c) 2006-2007, Ext JS, LLC.
4604  *
4605  * Originally Released Under LGPL - original licence link has changed is not relivant.
4606  *
4607  * Fork - LGPL
4608  * <script type="text/javascript">
4609  */
4610  
4611 /**
4612 * @class Roo.Template
4613 * Represents an HTML fragment template. Templates can be precompiled for greater performance.
4614 * For a list of available format functions, see {@link Roo.util.Format}.<br />
4615 * Usage:
4616 <pre><code>
4617 var t = new Roo.Template({
4618     html :  '&lt;div name="{id}"&gt;' + 
4619         '&lt;span class="{cls}"&gt;{name:trim} {someval:this.myformat}{value:ellipsis(10)}&lt;/span&gt;' +
4620         '&lt;/div&gt;',
4621     myformat: function (value, allValues) {
4622         return 'XX' + value;
4623     }
4624 });
4625 t.append('some-element', {id: 'myid', cls: 'myclass', name: 'foo', value: 'bar'});
4626 </code></pre>
4627 * For more information see this blog post with examples:
4628 *  <a href="http://www.cnitblog.com/seeyeah/archive/2011/12/30/38728.html/">DomHelper
4629      - Create Elements using DOM, HTML fragments and Templates</a>. 
4630 * @constructor
4631 * @param {Object} cfg - Configuration object.
4632 */
4633 Roo.Template = function(cfg){
4634     // BC!
4635     if(cfg instanceof Array){
4636         cfg = cfg.join("");
4637     }else if(arguments.length > 1){
4638         cfg = Array.prototype.join.call(arguments, "");
4639     }
4640     
4641     
4642     if (typeof(cfg) == 'object') {
4643         Roo.apply(this,cfg)
4644     } else {
4645         // bc
4646         this.html = cfg;
4647     }
4648     if (this.url) {
4649         this.load();
4650     }
4651     
4652 };
4653 Roo.Template.prototype = {
4654     
4655     /**
4656      * @cfg {String} url  The Url to load the template from. beware if you are loading from a url, the data may not be ready if you use it instantly..
4657      *                    it should be fixed so that template is observable...
4658      */
4659     url : false,
4660     /**
4661      * @cfg {String} html  The HTML fragment or an array of fragments to join("") or multiple arguments to join("")
4662      */
4663     html : '',
4664     /**
4665      * Returns an HTML fragment of this template with the specified values applied.
4666      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4667      * @return {String} The HTML fragment
4668      */
4669     applyTemplate : function(values){
4670         Roo.log(["applyTemplate", values]);
4671         try {
4672            
4673             if(this.compiled){
4674                 return this.compiled(values);
4675             }
4676             var useF = this.disableFormats !== true;
4677             var fm = Roo.util.Format, tpl = this;
4678             var fn = function(m, name, format, args){
4679                 if(format && useF){
4680                     if(format.substr(0, 5) == "this."){
4681                         return tpl.call(format.substr(5), values[name], values);
4682                     }else{
4683                         if(args){
4684                             // quoted values are required for strings in compiled templates, 
4685                             // but for non compiled we need to strip them
4686                             // quoted reversed for jsmin
4687                             var re = /^\s*['"](.*)["']\s*$/;
4688                             args = args.split(',');
4689                             for(var i = 0, len = args.length; i < len; i++){
4690                                 args[i] = args[i].replace(re, "$1");
4691                             }
4692                             args = [values[name]].concat(args);
4693                         }else{
4694                             args = [values[name]];
4695                         }
4696                         return fm[format].apply(fm, args);
4697                     }
4698                 }else{
4699                     return values[name] !== undefined ? values[name] : "";
4700                 }
4701             };
4702             return this.html.replace(this.re, fn);
4703         } catch (e) {
4704             Roo.log(e);
4705             throw e;
4706         }
4707          
4708     },
4709     
4710     loading : false,
4711       
4712     load : function ()
4713     {
4714          
4715         if (this.loading) {
4716             return;
4717         }
4718         var _t = this;
4719         
4720         this.loading = true;
4721         this.compiled = false;
4722         
4723         var cx = new Roo.data.Connection();
4724         cx.request({
4725             url : this.url,
4726             method : 'GET',
4727             success : function (response) {
4728                 _t.loading = false;
4729                 _t.html = response.responseText;
4730                 _t.url = false;
4731                 _t.compile();
4732              },
4733             failure : function(response) {
4734                 Roo.log("Template failed to load from " + _t.url);
4735                 _t.loading = false;
4736             }
4737         });
4738     },
4739
4740     /**
4741      * Sets the HTML used as the template and optionally compiles it.
4742      * @param {String} html
4743      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4744      * @return {Roo.Template} this
4745      */
4746     set : function(html, compile){
4747         this.html = html;
4748         this.compiled = null;
4749         if(compile){
4750             this.compile();
4751         }
4752         return this;
4753     },
4754     
4755     /**
4756      * True to disable format functions (defaults to false)
4757      * @type Boolean
4758      */
4759     disableFormats : false,
4760     
4761     /**
4762     * The regular expression used to match template variables 
4763     * @type RegExp
4764     * @property 
4765     */
4766     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4767     
4768     /**
4769      * Compiles the template into an internal function, eliminating the RegEx overhead.
4770      * @return {Roo.Template} this
4771      */
4772     compile : function(){
4773         var fm = Roo.util.Format;
4774         var useF = this.disableFormats !== true;
4775         var sep = Roo.isGecko ? "+" : ",";
4776         var fn = function(m, name, format, args){
4777             if(format && useF){
4778                 args = args ? ',' + args : "";
4779                 if(format.substr(0, 5) != "this."){
4780                     format = "fm." + format + '(';
4781                 }else{
4782                     format = 'this.call("'+ format.substr(5) + '", ';
4783                     args = ", values";
4784                 }
4785             }else{
4786                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4787             }
4788             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4789         };
4790         var body;
4791         // branched to use + in gecko and [].join() in others
4792         if(Roo.isGecko){
4793             body = "this.compiled = function(values){ return '" +
4794                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4795                     "';};";
4796         }else{
4797             body = ["this.compiled = function(values){ return ['"];
4798             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4799             body.push("'].join('');};");
4800             body = body.join('');
4801         }
4802         /**
4803          * eval:var:values
4804          * eval:var:fm
4805          */
4806         eval(body);
4807         return this;
4808     },
4809     
4810     // private function used to call members
4811     call : function(fnName, value, allValues){
4812         return this[fnName](value, allValues);
4813     },
4814     
4815     /**
4816      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4817      * @param {String/HTMLElement/Roo.Element} el The context element
4818      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4819      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4820      * @return {HTMLElement/Roo.Element} The new node or Element
4821      */
4822     insertFirst: function(el, values, returnElement){
4823         return this.doInsert('afterBegin', el, values, returnElement);
4824     },
4825
4826     /**
4827      * Applies the supplied values to the template and inserts the new node(s) before el.
4828      * @param {String/HTMLElement/Roo.Element} el The context element
4829      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4830      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4831      * @return {HTMLElement/Roo.Element} The new node or Element
4832      */
4833     insertBefore: function(el, values, returnElement){
4834         return this.doInsert('beforeBegin', el, values, returnElement);
4835     },
4836
4837     /**
4838      * Applies the supplied values to the template and inserts the new node(s) after el.
4839      * @param {String/HTMLElement/Roo.Element} el The context element
4840      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4841      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4842      * @return {HTMLElement/Roo.Element} The new node or Element
4843      */
4844     insertAfter : function(el, values, returnElement){
4845         return this.doInsert('afterEnd', el, values, returnElement);
4846     },
4847     
4848     /**
4849      * Applies the supplied values to the template and appends the new node(s) to el.
4850      * @param {String/HTMLElement/Roo.Element} el The context element
4851      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4852      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4853      * @return {HTMLElement/Roo.Element} The new node or Element
4854      */
4855     append : function(el, values, returnElement){
4856         return this.doInsert('beforeEnd', el, values, returnElement);
4857     },
4858
4859     doInsert : function(where, el, values, returnEl){
4860         el = Roo.getDom(el);
4861         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4862         return returnEl ? Roo.get(newNode, true) : newNode;
4863     },
4864
4865     /**
4866      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4867      * @param {String/HTMLElement/Roo.Element} el The context element
4868      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4869      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4870      * @return {HTMLElement/Roo.Element} The new node or Element
4871      */
4872     overwrite : function(el, values, returnElement){
4873         el = Roo.getDom(el);
4874         el.innerHTML = this.applyTemplate(values);
4875         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4876     }
4877 };
4878 /**
4879  * Alias for {@link #applyTemplate}
4880  * @method
4881  */
4882 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4883
4884 // backwards compat
4885 Roo.DomHelper.Template = Roo.Template;
4886
4887 /**
4888  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4889  * @param {String/HTMLElement} el A DOM element or its id
4890  * @returns {Roo.Template} The created template
4891  * @static
4892  */
4893 Roo.Template.from = function(el){
4894     el = Roo.getDom(el);
4895     return new Roo.Template(el.value || el.innerHTML);
4896 };/*
4897  * Based on:
4898  * Ext JS Library 1.1.1
4899  * Copyright(c) 2006-2007, Ext JS, LLC.
4900  *
4901  * Originally Released Under LGPL - original licence link has changed is not relivant.
4902  *
4903  * Fork - LGPL
4904  * <script type="text/javascript">
4905  */
4906  
4907
4908 /*
4909  * This is code is also distributed under MIT license for use
4910  * with jQuery and prototype JavaScript libraries.
4911  */
4912 /**
4913  * @class Roo.DomQuery
4914 Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
4915 <p>
4916 DomQuery supports most of the <a href="http://www.w3.org/TR/2005/WD-css3-selectors-20051215/">CSS3 selectors spec</a>, along with some custom selectors and basic XPath.</p>
4917
4918 <p>
4919 All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example "div.foo:nth-child(odd)[@foo=bar].bar:first" would be a perfectly valid selector. Node filters are processed in the order in which they appear, which allows you to optimize your queries for your document structure.
4920 </p>
4921 <h4>Element Selectors:</h4>
4922 <ul class="list">
4923     <li> <b>*</b> any element</li>
4924     <li> <b>E</b> an element with the tag E</li>
4925     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4926     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4927     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4928     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4929 </ul>
4930 <h4>Attribute Selectors:</h4>
4931 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4932 <ul class="list">
4933     <li> <b>E[foo]</b> has an attribute "foo"</li>
4934     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4935     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4936     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4937     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4938     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4939     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4940 </ul>
4941 <h4>Pseudo Classes:</h4>
4942 <ul class="list">
4943     <li> <b>E:first-child</b> E is the first child of its parent</li>
4944     <li> <b>E:last-child</b> E is the last child of its parent</li>
4945     <li> <b>E:nth-child(<i>n</i>)</b> E is the <i>n</i>th child of its parent (1 based as per the spec)</li>
4946     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4947     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4948     <li> <b>E:only-child</b> E is the only child of its parent</li>
4949     <li> <b>E:checked</b> E is an element that is has a checked attribute that is true (e.g. a radio or checkbox) </li>
4950     <li> <b>E:first</b> the first E in the resultset</li>
4951     <li> <b>E:last</b> the last E in the resultset</li>
4952     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4953     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4954     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4955     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4956     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4957     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4958     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4959     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4960     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4961 </ul>
4962 <h4>CSS Value Selectors:</h4>
4963 <ul class="list">
4964     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4965     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4966     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4967     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4968     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4969     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4970 </ul>
4971  * @singleton
4972  */
4973 Roo.DomQuery = function(){
4974     var cache = {}, simpleCache = {}, valueCache = {};
4975     var nonSpace = /\S/;
4976     var trimRe = /^\s+|\s+$/g;
4977     var tplRe = /\{(\d+)\}/g;
4978     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4979     var tagTokenRe = /^(#)?([\w-\*]+)/;
4980     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4981
4982     function child(p, index){
4983         var i = 0;
4984         var n = p.firstChild;
4985         while(n){
4986             if(n.nodeType == 1){
4987                if(++i == index){
4988                    return n;
4989                }
4990             }
4991             n = n.nextSibling;
4992         }
4993         return null;
4994     };
4995
4996     function next(n){
4997         while((n = n.nextSibling) && n.nodeType != 1);
4998         return n;
4999     };
5000
5001     function prev(n){
5002         while((n = n.previousSibling) && n.nodeType != 1);
5003         return n;
5004     };
5005
5006     function children(d){
5007         var n = d.firstChild, ni = -1;
5008             while(n){
5009                 var nx = n.nextSibling;
5010                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5011                     d.removeChild(n);
5012                 }else{
5013                     n.nodeIndex = ++ni;
5014                 }
5015                 n = nx;
5016             }
5017             return this;
5018         };
5019
5020     function byClassName(c, a, v){
5021         if(!v){
5022             return c;
5023         }
5024         var r = [], ri = -1, cn;
5025         for(var i = 0, ci; ci = c[i]; i++){
5026             if((' '+ci.className+' ').indexOf(v) != -1){
5027                 r[++ri] = ci;
5028             }
5029         }
5030         return r;
5031     };
5032
5033     function attrValue(n, attr){
5034         if(!n.tagName && typeof n.length != "undefined"){
5035             n = n[0];
5036         }
5037         if(!n){
5038             return null;
5039         }
5040         if(attr == "for"){
5041             return n.htmlFor;
5042         }
5043         if(attr == "class" || attr == "className"){
5044             return n.className;
5045         }
5046         return n.getAttribute(attr) || n[attr];
5047
5048     };
5049
5050     function getNodes(ns, mode, tagName){
5051         var result = [], ri = -1, cs;
5052         if(!ns){
5053             return result;
5054         }
5055         tagName = tagName || "*";
5056         if(typeof ns.getElementsByTagName != "undefined"){
5057             ns = [ns];
5058         }
5059         if(!mode){
5060             for(var i = 0, ni; ni = ns[i]; i++){
5061                 cs = ni.getElementsByTagName(tagName);
5062                 for(var j = 0, ci; ci = cs[j]; j++){
5063                     result[++ri] = ci;
5064                 }
5065             }
5066         }else if(mode == "/" || mode == ">"){
5067             var utag = tagName.toUpperCase();
5068             for(var i = 0, ni, cn; ni = ns[i]; i++){
5069                 cn = ni.children || ni.childNodes;
5070                 for(var j = 0, cj; cj = cn[j]; j++){
5071                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5072                         result[++ri] = cj;
5073                     }
5074                 }
5075             }
5076         }else if(mode == "+"){
5077             var utag = tagName.toUpperCase();
5078             for(var i = 0, n; n = ns[i]; i++){
5079                 while((n = n.nextSibling) && n.nodeType != 1);
5080                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5081                     result[++ri] = n;
5082                 }
5083             }
5084         }else if(mode == "~"){
5085             for(var i = 0, n; n = ns[i]; i++){
5086                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5087                 if(n){
5088                     result[++ri] = n;
5089                 }
5090             }
5091         }
5092         return result;
5093     };
5094
5095     function concat(a, b){
5096         if(b.slice){
5097             return a.concat(b);
5098         }
5099         for(var i = 0, l = b.length; i < l; i++){
5100             a[a.length] = b[i];
5101         }
5102         return a;
5103     }
5104
5105     function byTag(cs, tagName){
5106         if(cs.tagName || cs == document){
5107             cs = [cs];
5108         }
5109         if(!tagName){
5110             return cs;
5111         }
5112         var r = [], ri = -1;
5113         tagName = tagName.toLowerCase();
5114         for(var i = 0, ci; ci = cs[i]; i++){
5115             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5116                 r[++ri] = ci;
5117             }
5118         }
5119         return r;
5120     };
5121
5122     function byId(cs, attr, id){
5123         if(cs.tagName || cs == document){
5124             cs = [cs];
5125         }
5126         if(!id){
5127             return cs;
5128         }
5129         var r = [], ri = -1;
5130         for(var i = 0,ci; ci = cs[i]; i++){
5131             if(ci && ci.id == id){
5132                 r[++ri] = ci;
5133                 return r;
5134             }
5135         }
5136         return r;
5137     };
5138
5139     function byAttribute(cs, attr, value, op, custom){
5140         var r = [], ri = -1, st = custom=="{";
5141         var f = Roo.DomQuery.operators[op];
5142         for(var i = 0, ci; ci = cs[i]; i++){
5143             var a;
5144             if(st){
5145                 a = Roo.DomQuery.getStyle(ci, attr);
5146             }
5147             else if(attr == "class" || attr == "className"){
5148                 a = ci.className;
5149             }else if(attr == "for"){
5150                 a = ci.htmlFor;
5151             }else if(attr == "href"){
5152                 a = ci.getAttribute("href", 2);
5153             }else{
5154                 a = ci.getAttribute(attr);
5155             }
5156             if((f && f(a, value)) || (!f && a)){
5157                 r[++ri] = ci;
5158             }
5159         }
5160         return r;
5161     };
5162
5163     function byPseudo(cs, name, value){
5164         return Roo.DomQuery.pseudos[name](cs, value);
5165     };
5166
5167     // This is for IE MSXML which does not support expandos.
5168     // IE runs the same speed using setAttribute, however FF slows way down
5169     // and Safari completely fails so they need to continue to use expandos.
5170     var isIE = window.ActiveXObject ? true : false;
5171
5172     // this eval is stop the compressor from
5173     // renaming the variable to something shorter
5174     
5175     /** eval:var:batch */
5176     var batch = 30803; 
5177
5178     var key = 30803;
5179
5180     function nodupIEXml(cs){
5181         var d = ++key;
5182         cs[0].setAttribute("_nodup", d);
5183         var r = [cs[0]];
5184         for(var i = 1, len = cs.length; i < len; i++){
5185             var c = cs[i];
5186             if(!c.getAttribute("_nodup") != d){
5187                 c.setAttribute("_nodup", d);
5188                 r[r.length] = c;
5189             }
5190         }
5191         for(var i = 0, len = cs.length; i < len; i++){
5192             cs[i].removeAttribute("_nodup");
5193         }
5194         return r;
5195     }
5196
5197     function nodup(cs){
5198         if(!cs){
5199             return [];
5200         }
5201         var len = cs.length, c, i, r = cs, cj, ri = -1;
5202         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5203             return cs;
5204         }
5205         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5206             return nodupIEXml(cs);
5207         }
5208         var d = ++key;
5209         cs[0]._nodup = d;
5210         for(i = 1; c = cs[i]; i++){
5211             if(c._nodup != d){
5212                 c._nodup = d;
5213             }else{
5214                 r = [];
5215                 for(var j = 0; j < i; j++){
5216                     r[++ri] = cs[j];
5217                 }
5218                 for(j = i+1; cj = cs[j]; j++){
5219                     if(cj._nodup != d){
5220                         cj._nodup = d;
5221                         r[++ri] = cj;
5222                     }
5223                 }
5224                 return r;
5225             }
5226         }
5227         return r;
5228     }
5229
5230     function quickDiffIEXml(c1, c2){
5231         var d = ++key;
5232         for(var i = 0, len = c1.length; i < len; i++){
5233             c1[i].setAttribute("_qdiff", d);
5234         }
5235         var r = [];
5236         for(var i = 0, len = c2.length; i < len; i++){
5237             if(c2[i].getAttribute("_qdiff") != d){
5238                 r[r.length] = c2[i];
5239             }
5240         }
5241         for(var i = 0, len = c1.length; i < len; i++){
5242            c1[i].removeAttribute("_qdiff");
5243         }
5244         return r;
5245     }
5246
5247     function quickDiff(c1, c2){
5248         var len1 = c1.length;
5249         if(!len1){
5250             return c2;
5251         }
5252         if(isIE && c1[0].selectSingleNode){
5253             return quickDiffIEXml(c1, c2);
5254         }
5255         var d = ++key;
5256         for(var i = 0; i < len1; i++){
5257             c1[i]._qdiff = d;
5258         }
5259         var r = [];
5260         for(var i = 0, len = c2.length; i < len; i++){
5261             if(c2[i]._qdiff != d){
5262                 r[r.length] = c2[i];
5263             }
5264         }
5265         return r;
5266     }
5267
5268     function quickId(ns, mode, root, id){
5269         if(ns == root){
5270            var d = root.ownerDocument || root;
5271            return d.getElementById(id);
5272         }
5273         ns = getNodes(ns, mode, "*");
5274         return byId(ns, null, id);
5275     }
5276
5277     return {
5278         getStyle : function(el, name){
5279             return Roo.fly(el).getStyle(name);
5280         },
5281         /**
5282          * Compiles a selector/xpath query into a reusable function. The returned function
5283          * takes one parameter "root" (optional), which is the context node from where the query should start.
5284          * @param {String} selector The selector/xpath query
5285          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5286          * @return {Function}
5287          */
5288         compile : function(path, type){
5289             type = type || "select";
5290             
5291             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5292             var q = path, mode, lq;
5293             var tk = Roo.DomQuery.matchers;
5294             var tklen = tk.length;
5295             var mm;
5296
5297             // accept leading mode switch
5298             var lmode = q.match(modeRe);
5299             if(lmode && lmode[1]){
5300                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5301                 q = q.replace(lmode[1], "");
5302             }
5303             // strip leading slashes
5304             while(path.substr(0, 1)=="/"){
5305                 path = path.substr(1);
5306             }
5307
5308             while(q && lq != q){
5309                 lq = q;
5310                 var tm = q.match(tagTokenRe);
5311                 if(type == "select"){
5312                     if(tm){
5313                         if(tm[1] == "#"){
5314                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5315                         }else{
5316                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5317                         }
5318                         q = q.replace(tm[0], "");
5319                     }else if(q.substr(0, 1) != '@'){
5320                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5321                     }
5322                 }else{
5323                     if(tm){
5324                         if(tm[1] == "#"){
5325                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5326                         }else{
5327                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5328                         }
5329                         q = q.replace(tm[0], "");
5330                     }
5331                 }
5332                 while(!(mm = q.match(modeRe))){
5333                     var matched = false;
5334                     for(var j = 0; j < tklen; j++){
5335                         var t = tk[j];
5336                         var m = q.match(t.re);
5337                         if(m){
5338                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5339                                                     return m[i];
5340                                                 });
5341                             q = q.replace(m[0], "");
5342                             matched = true;
5343                             break;
5344                         }
5345                     }
5346                     // prevent infinite loop on bad selector
5347                     if(!matched){
5348                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5349                     }
5350                 }
5351                 if(mm[1]){
5352                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5353                     q = q.replace(mm[1], "");
5354                 }
5355             }
5356             fn[fn.length] = "return nodup(n);\n}";
5357             
5358              /** 
5359               * list of variables that need from compression as they are used by eval.
5360              *  eval:var:batch 
5361              *  eval:var:nodup
5362              *  eval:var:byTag
5363              *  eval:var:ById
5364              *  eval:var:getNodes
5365              *  eval:var:quickId
5366              *  eval:var:mode
5367              *  eval:var:root
5368              *  eval:var:n
5369              *  eval:var:byClassName
5370              *  eval:var:byPseudo
5371              *  eval:var:byAttribute
5372              *  eval:var:attrValue
5373              * 
5374              **/ 
5375             eval(fn.join(""));
5376             return f;
5377         },
5378
5379         /**
5380          * Selects a group of elements.
5381          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5382          * @param {Node} root (optional) The start of the query (defaults to document).
5383          * @return {Array}
5384          */
5385         select : function(path, root, type){
5386             if(!root || root == document){
5387                 root = document;
5388             }
5389             if(typeof root == "string"){
5390                 root = document.getElementById(root);
5391             }
5392             var paths = path.split(",");
5393             var results = [];
5394             for(var i = 0, len = paths.length; i < len; i++){
5395                 var p = paths[i].replace(trimRe, "");
5396                 if(!cache[p]){
5397                     cache[p] = Roo.DomQuery.compile(p);
5398                     if(!cache[p]){
5399                         throw p + " is not a valid selector";
5400                     }
5401                 }
5402                 var result = cache[p](root);
5403                 if(result && result != document){
5404                     results = results.concat(result);
5405                 }
5406             }
5407             if(paths.length > 1){
5408                 return nodup(results);
5409             }
5410             return results;
5411         },
5412
5413         /**
5414          * Selects a single element.
5415          * @param {String} selector The selector/xpath query
5416          * @param {Node} root (optional) The start of the query (defaults to document).
5417          * @return {Element}
5418          */
5419         selectNode : function(path, root){
5420             return Roo.DomQuery.select(path, root)[0];
5421         },
5422
5423         /**
5424          * Selects the value of a node, optionally replacing null with the defaultValue.
5425          * @param {String} selector The selector/xpath query
5426          * @param {Node} root (optional) The start of the query (defaults to document).
5427          * @param {String} defaultValue
5428          */
5429         selectValue : function(path, root, defaultValue){
5430             path = path.replace(trimRe, "");
5431             if(!valueCache[path]){
5432                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5433             }
5434             var n = valueCache[path](root);
5435             n = n[0] ? n[0] : n;
5436             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5437             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5438         },
5439
5440         /**
5441          * Selects the value of a node, parsing integers and floats.
5442          * @param {String} selector The selector/xpath query
5443          * @param {Node} root (optional) The start of the query (defaults to document).
5444          * @param {Number} defaultValue
5445          * @return {Number}
5446          */
5447         selectNumber : function(path, root, defaultValue){
5448             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5449             return parseFloat(v);
5450         },
5451
5452         /**
5453          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5454          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5455          * @param {String} selector The simple selector to test
5456          * @return {Boolean}
5457          */
5458         is : function(el, ss){
5459             if(typeof el == "string"){
5460                 el = document.getElementById(el);
5461             }
5462             var isArray = (el instanceof Array);
5463             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5464             return isArray ? (result.length == el.length) : (result.length > 0);
5465         },
5466
5467         /**
5468          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5469          * @param {Array} el An array of elements to filter
5470          * @param {String} selector The simple selector to test
5471          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5472          * the selector instead of the ones that match
5473          * @return {Array}
5474          */
5475         filter : function(els, ss, nonMatches){
5476             ss = ss.replace(trimRe, "");
5477             if(!simpleCache[ss]){
5478                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5479             }
5480             var result = simpleCache[ss](els);
5481             return nonMatches ? quickDiff(result, els) : result;
5482         },
5483
5484         /**
5485          * Collection of matching regular expressions and code snippets.
5486          */
5487         matchers : [{
5488                 re: /^\.([\w-]+)/,
5489                 select: 'n = byClassName(n, null, " {1} ");'
5490             }, {
5491                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5492                 select: 'n = byPseudo(n, "{1}", "{2}");'
5493             },{
5494                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5495                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5496             }, {
5497                 re: /^#([\w-]+)/,
5498                 select: 'n = byId(n, null, "{1}");'
5499             },{
5500                 re: /^@([\w-]+)/,
5501                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5502             }
5503         ],
5504
5505         /**
5506          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5507          * New operators can be added as long as the match the format <i>c</i>= where <i>c</i> is any character other than space, &gt; &lt;.
5508          */
5509         operators : {
5510             "=" : function(a, v){
5511                 return a == v;
5512             },
5513             "!=" : function(a, v){
5514                 return a != v;
5515             },
5516             "^=" : function(a, v){
5517                 return a && a.substr(0, v.length) == v;
5518             },
5519             "$=" : function(a, v){
5520                 return a && a.substr(a.length-v.length) == v;
5521             },
5522             "*=" : function(a, v){
5523                 return a && a.indexOf(v) !== -1;
5524             },
5525             "%=" : function(a, v){
5526                 return (a % v) == 0;
5527             },
5528             "|=" : function(a, v){
5529                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5530             },
5531             "~=" : function(a, v){
5532                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5533             }
5534         },
5535
5536         /**
5537          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5538          * and the argument (if any) supplied in the selector.
5539          */
5540         pseudos : {
5541             "first-child" : function(c){
5542                 var r = [], ri = -1, n;
5543                 for(var i = 0, ci; ci = n = c[i]; i++){
5544                     while((n = n.previousSibling) && n.nodeType != 1);
5545                     if(!n){
5546                         r[++ri] = ci;
5547                     }
5548                 }
5549                 return r;
5550             },
5551
5552             "last-child" : function(c){
5553                 var r = [], ri = -1, n;
5554                 for(var i = 0, ci; ci = n = c[i]; i++){
5555                     while((n = n.nextSibling) && n.nodeType != 1);
5556                     if(!n){
5557                         r[++ri] = ci;
5558                     }
5559                 }
5560                 return r;
5561             },
5562
5563             "nth-child" : function(c, a) {
5564                 var r = [], ri = -1;
5565                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5566                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5567                 for(var i = 0, n; n = c[i]; i++){
5568                     var pn = n.parentNode;
5569                     if (batch != pn._batch) {
5570                         var j = 0;
5571                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5572                             if(cn.nodeType == 1){
5573                                cn.nodeIndex = ++j;
5574                             }
5575                         }
5576                         pn._batch = batch;
5577                     }
5578                     if (f == 1) {
5579                         if (l == 0 || n.nodeIndex == l){
5580                             r[++ri] = n;
5581                         }
5582                     } else if ((n.nodeIndex + l) % f == 0){
5583                         r[++ri] = n;
5584                     }
5585                 }
5586
5587                 return r;
5588             },
5589
5590             "only-child" : function(c){
5591                 var r = [], ri = -1;;
5592                 for(var i = 0, ci; ci = c[i]; i++){
5593                     if(!prev(ci) && !next(ci)){
5594                         r[++ri] = ci;
5595                     }
5596                 }
5597                 return r;
5598             },
5599
5600             "empty" : function(c){
5601                 var r = [], ri = -1;
5602                 for(var i = 0, ci; ci = c[i]; i++){
5603                     var cns = ci.childNodes, j = 0, cn, empty = true;
5604                     while(cn = cns[j]){
5605                         ++j;
5606                         if(cn.nodeType == 1 || cn.nodeType == 3){
5607                             empty = false;
5608                             break;
5609                         }
5610                     }
5611                     if(empty){
5612                         r[++ri] = ci;
5613                     }
5614                 }
5615                 return r;
5616             },
5617
5618             "contains" : function(c, v){
5619                 var r = [], ri = -1;
5620                 for(var i = 0, ci; ci = c[i]; i++){
5621                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5622                         r[++ri] = ci;
5623                     }
5624                 }
5625                 return r;
5626             },
5627
5628             "nodeValue" : function(c, v){
5629                 var r = [], ri = -1;
5630                 for(var i = 0, ci; ci = c[i]; i++){
5631                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5632                         r[++ri] = ci;
5633                     }
5634                 }
5635                 return r;
5636             },
5637
5638             "checked" : function(c){
5639                 var r = [], ri = -1;
5640                 for(var i = 0, ci; ci = c[i]; i++){
5641                     if(ci.checked == true){
5642                         r[++ri] = ci;
5643                     }
5644                 }
5645                 return r;
5646             },
5647
5648             "not" : function(c, ss){
5649                 return Roo.DomQuery.filter(c, ss, true);
5650             },
5651
5652             "odd" : function(c){
5653                 return this["nth-child"](c, "odd");
5654             },
5655
5656             "even" : function(c){
5657                 return this["nth-child"](c, "even");
5658             },
5659
5660             "nth" : function(c, a){
5661                 return c[a-1] || [];
5662             },
5663
5664             "first" : function(c){
5665                 return c[0] || [];
5666             },
5667
5668             "last" : function(c){
5669                 return c[c.length-1] || [];
5670             },
5671
5672             "has" : function(c, ss){
5673                 var s = Roo.DomQuery.select;
5674                 var r = [], ri = -1;
5675                 for(var i = 0, ci; ci = c[i]; i++){
5676                     if(s(ss, ci).length > 0){
5677                         r[++ri] = ci;
5678                     }
5679                 }
5680                 return r;
5681             },
5682
5683             "next" : function(c, ss){
5684                 var is = Roo.DomQuery.is;
5685                 var r = [], ri = -1;
5686                 for(var i = 0, ci; ci = c[i]; i++){
5687                     var n = next(ci);
5688                     if(n && is(n, ss)){
5689                         r[++ri] = ci;
5690                     }
5691                 }
5692                 return r;
5693             },
5694
5695             "prev" : function(c, ss){
5696                 var is = Roo.DomQuery.is;
5697                 var r = [], ri = -1;
5698                 for(var i = 0, ci; ci = c[i]; i++){
5699                     var n = prev(ci);
5700                     if(n && is(n, ss)){
5701                         r[++ri] = ci;
5702                     }
5703                 }
5704                 return r;
5705             }
5706         }
5707     };
5708 }();
5709
5710 /**
5711  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5712  * @param {String} path The selector/xpath query
5713  * @param {Node} root (optional) The start of the query (defaults to document).
5714  * @return {Array}
5715  * @member Roo
5716  * @method query
5717  */
5718 Roo.query = Roo.DomQuery.select;
5719 /*
5720  * Based on:
5721  * Ext JS Library 1.1.1
5722  * Copyright(c) 2006-2007, Ext JS, LLC.
5723  *
5724  * Originally Released Under LGPL - original licence link has changed is not relivant.
5725  *
5726  * Fork - LGPL
5727  * <script type="text/javascript">
5728  */
5729
5730 /**
5731  * @class Roo.util.Observable
5732  * Base class that provides a common interface for publishing events. Subclasses are expected to
5733  * to have a property "events" with all the events defined.<br>
5734  * For example:
5735  * <pre><code>
5736  Employee = function(name){
5737     this.name = name;
5738     this.addEvents({
5739         "fired" : true,
5740         "quit" : true
5741     });
5742  }
5743  Roo.extend(Employee, Roo.util.Observable);
5744 </code></pre>
5745  * @param {Object} config properties to use (incuding events / listeners)
5746  */
5747
5748 Roo.util.Observable = function(cfg){
5749     
5750     cfg = cfg|| {};
5751     this.addEvents(cfg.events || {});
5752     if (cfg.events) {
5753         delete cfg.events; // make sure
5754     }
5755      
5756     Roo.apply(this, cfg);
5757     
5758     if(this.listeners){
5759         this.on(this.listeners);
5760         delete this.listeners;
5761     }
5762 };
5763 Roo.util.Observable.prototype = {
5764     /** 
5765  * @cfg {Object} listeners  list of events and functions to call for this object, 
5766  * For example :
5767  * <pre><code>
5768     listeners :  { 
5769        'click' : function(e) {
5770            ..... 
5771         } ,
5772         .... 
5773     } 
5774   </code></pre>
5775  */
5776     
5777     
5778     /**
5779      * Fires the specified event with the passed parameters (minus the event name).
5780      * @param {String} eventName
5781      * @param {Object...} args Variable number of parameters are passed to handlers
5782      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5783      */
5784     fireEvent : function(){
5785         var ce = this.events[arguments[0].toLowerCase()];
5786         if(typeof ce == "object"){
5787             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5788         }else{
5789             return true;
5790         }
5791     },
5792
5793     // private
5794     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5795
5796     /**
5797      * Appends an event handler to this component
5798      * @param {String}   eventName The type of event to listen for
5799      * @param {Function} handler The method the event invokes
5800      * @param {Object}   scope (optional) The scope in which to execute the handler
5801      * function. The handler function's "this" context.
5802      * @param {Object}   options (optional) An object containing handler configuration
5803      * properties. This may contain any of the following properties:<ul>
5804      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5805      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5806      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5807      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5808      * by the specified number of milliseconds. If the event fires again within that time, the original
5809      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5810      * </ul><br>
5811      * <p>
5812      * <b>Combining Options</b><br>
5813      * Using the options argument, it is possible to combine different types of listeners:<br>
5814      * <br>
5815      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5816                 <pre><code>
5817                 el.on('click', this.onClick, this, {
5818                         single: true,
5819                 delay: 100,
5820                 forumId: 4
5821                 });
5822                 </code></pre>
5823      * <p>
5824      * <b>Attaching multiple handlers in 1 call</b><br>
5825      * The method also allows for a single argument to be passed which is a config object containing properties
5826      * which specify multiple handlers.
5827      * <pre><code>
5828                 el.on({
5829                         'click': {
5830                         fn: this.onClick,
5831                         scope: this,
5832                         delay: 100
5833                 }, 
5834                 'mouseover': {
5835                         fn: this.onMouseOver,
5836                         scope: this
5837                 },
5838                 'mouseout': {
5839                         fn: this.onMouseOut,
5840                         scope: this
5841                 }
5842                 });
5843                 </code></pre>
5844      * <p>
5845      * Or a shorthand syntax which passes the same scope object to all handlers:
5846         <pre><code>
5847                 el.on({
5848                         'click': this.onClick,
5849                 'mouseover': this.onMouseOver,
5850                 'mouseout': this.onMouseOut,
5851                 scope: this
5852                 });
5853                 </code></pre>
5854      */
5855     addListener : function(eventName, fn, scope, o){
5856         if(typeof eventName == "object"){
5857             o = eventName;
5858             for(var e in o){
5859                 if(this.filterOptRe.test(e)){
5860                     continue;
5861                 }
5862                 if(typeof o[e] == "function"){
5863                     // shared options
5864                     this.addListener(e, o[e], o.scope,  o);
5865                 }else{
5866                     // individual options
5867                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5868                 }
5869             }
5870             return;
5871         }
5872         o = (!o || typeof o == "boolean") ? {} : o;
5873         eventName = eventName.toLowerCase();
5874         var ce = this.events[eventName] || true;
5875         if(typeof ce == "boolean"){
5876             ce = new Roo.util.Event(this, eventName);
5877             this.events[eventName] = ce;
5878         }
5879         ce.addListener(fn, scope, o);
5880     },
5881
5882     /**
5883      * Removes a listener
5884      * @param {String}   eventName     The type of event to listen for
5885      * @param {Function} handler        The handler to remove
5886      * @param {Object}   scope  (optional) The scope (this object) for the handler
5887      */
5888     removeListener : function(eventName, fn, scope){
5889         var ce = this.events[eventName.toLowerCase()];
5890         if(typeof ce == "object"){
5891             ce.removeListener(fn, scope);
5892         }
5893     },
5894
5895     /**
5896      * Removes all listeners for this object
5897      */
5898     purgeListeners : function(){
5899         for(var evt in this.events){
5900             if(typeof this.events[evt] == "object"){
5901                  this.events[evt].clearListeners();
5902             }
5903         }
5904     },
5905
5906     relayEvents : function(o, events){
5907         var createHandler = function(ename){
5908             return function(){
5909                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5910             };
5911         };
5912         for(var i = 0, len = events.length; i < len; i++){
5913             var ename = events[i];
5914             if(!this.events[ename]){ this.events[ename] = true; };
5915             o.on(ename, createHandler(ename), this);
5916         }
5917     },
5918
5919     /**
5920      * Used to define events on this Observable
5921      * @param {Object} object The object with the events defined
5922      */
5923     addEvents : function(o){
5924         if(!this.events){
5925             this.events = {};
5926         }
5927         Roo.applyIf(this.events, o);
5928     },
5929
5930     /**
5931      * Checks to see if this object has any listeners for a specified event
5932      * @param {String} eventName The name of the event to check for
5933      * @return {Boolean} True if the event is being listened for, else false
5934      */
5935     hasListener : function(eventName){
5936         var e = this.events[eventName];
5937         return typeof e == "object" && e.listeners.length > 0;
5938     }
5939 };
5940 /**
5941  * Appends an event handler to this element (shorthand for addListener)
5942  * @param {String}   eventName     The type of event to listen for
5943  * @param {Function} handler        The method the event invokes
5944  * @param {Object}   scope (optional) The scope in which to execute the handler
5945  * function. The handler function's "this" context.
5946  * @param {Object}   options  (optional)
5947  * @method
5948  */
5949 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5950 /**
5951  * Removes a listener (shorthand for removeListener)
5952  * @param {String}   eventName     The type of event to listen for
5953  * @param {Function} handler        The handler to remove
5954  * @param {Object}   scope  (optional) The scope (this object) for the handler
5955  * @method
5956  */
5957 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5958
5959 /**
5960  * Starts capture on the specified Observable. All events will be passed
5961  * to the supplied function with the event name + standard signature of the event
5962  * <b>before</b> the event is fired. If the supplied function returns false,
5963  * the event will not fire.
5964  * @param {Observable} o The Observable to capture
5965  * @param {Function} fn The function to call
5966  * @param {Object} scope (optional) The scope (this object) for the fn
5967  * @static
5968  */
5969 Roo.util.Observable.capture = function(o, fn, scope){
5970     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5971 };
5972
5973 /**
5974  * Removes <b>all</b> added captures from the Observable.
5975  * @param {Observable} o The Observable to release
5976  * @static
5977  */
5978 Roo.util.Observable.releaseCapture = function(o){
5979     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5980 };
5981
5982 (function(){
5983
5984     var createBuffered = function(h, o, scope){
5985         var task = new Roo.util.DelayedTask();
5986         return function(){
5987             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5988         };
5989     };
5990
5991     var createSingle = function(h, e, fn, scope){
5992         return function(){
5993             e.removeListener(fn, scope);
5994             return h.apply(scope, arguments);
5995         };
5996     };
5997
5998     var createDelayed = function(h, o, scope){
5999         return function(){
6000             var args = Array.prototype.slice.call(arguments, 0);
6001             setTimeout(function(){
6002                 h.apply(scope, args);
6003             }, o.delay || 10);
6004         };
6005     };
6006
6007     Roo.util.Event = function(obj, name){
6008         this.name = name;
6009         this.obj = obj;
6010         this.listeners = [];
6011     };
6012
6013     Roo.util.Event.prototype = {
6014         addListener : function(fn, scope, options){
6015             var o = options || {};
6016             scope = scope || this.obj;
6017             if(!this.isListening(fn, scope)){
6018                 var l = {fn: fn, scope: scope, options: o};
6019                 var h = fn;
6020                 if(o.delay){
6021                     h = createDelayed(h, o, scope);
6022                 }
6023                 if(o.single){
6024                     h = createSingle(h, this, fn, scope);
6025                 }
6026                 if(o.buffer){
6027                     h = createBuffered(h, o, scope);
6028                 }
6029                 l.fireFn = h;
6030                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6031                     this.listeners.push(l);
6032                 }else{
6033                     this.listeners = this.listeners.slice(0);
6034                     this.listeners.push(l);
6035                 }
6036             }
6037         },
6038
6039         findListener : function(fn, scope){
6040             scope = scope || this.obj;
6041             var ls = this.listeners;
6042             for(var i = 0, len = ls.length; i < len; i++){
6043                 var l = ls[i];
6044                 if(l.fn == fn && l.scope == scope){
6045                     return i;
6046                 }
6047             }
6048             return -1;
6049         },
6050
6051         isListening : function(fn, scope){
6052             return this.findListener(fn, scope) != -1;
6053         },
6054
6055         removeListener : function(fn, scope){
6056             var index;
6057             if((index = this.findListener(fn, scope)) != -1){
6058                 if(!this.firing){
6059                     this.listeners.splice(index, 1);
6060                 }else{
6061                     this.listeners = this.listeners.slice(0);
6062                     this.listeners.splice(index, 1);
6063                 }
6064                 return true;
6065             }
6066             return false;
6067         },
6068
6069         clearListeners : function(){
6070             this.listeners = [];
6071         },
6072
6073         fire : function(){
6074             var ls = this.listeners, scope, len = ls.length;
6075             if(len > 0){
6076                 this.firing = true;
6077                 
6078                 for(var i = 0; i < len; i++){
6079                     var args = Array.prototype.slice.call(arguments, 0);
6080                     var l = ls[i];
6081                     args.push(l.options);
6082                     if(l.fireFn.apply(l.scope||this.obj||window, args) === false){
6083                         this.firing = false;
6084                         return false;
6085                     }
6086                 }
6087                 this.firing = false;
6088             }
6089             return true;
6090         }
6091     };
6092 })();/*
6093  * RooJS Library 
6094  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6095  *
6096  * Licence LGPL 
6097  *
6098  */
6099  
6100 /**
6101  * @class Roo.Document
6102  * @extends Roo.util.Observable
6103  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6104  * 
6105  * @param {Object} config the methods and properties of the 'base' class for the application.
6106  * 
6107  *  Generic Page handler - implement this to start your app..
6108  * 
6109  * eg.
6110  *  MyProject = new Roo.Document({
6111         events : {
6112             'load' : true // your events..
6113         },
6114         listeners : {
6115             'ready' : function() {
6116                 // fired on Roo.onReady()
6117             }
6118         }
6119  * 
6120  */
6121 Roo.Document = function(cfg) {
6122      
6123     this.addEvents({ 
6124         'ready' : true
6125     });
6126     Roo.util.Observable.call(this,cfg);
6127     
6128     var _this = this;
6129     
6130     Roo.onReady(function() {
6131         _this.fireEvent('ready');
6132     },null,false);
6133     
6134     
6135 }
6136
6137 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6138  * Based on:
6139  * Ext JS Library 1.1.1
6140  * Copyright(c) 2006-2007, Ext JS, LLC.
6141  *
6142  * Originally Released Under LGPL - original licence link has changed is not relivant.
6143  *
6144  * Fork - LGPL
6145  * <script type="text/javascript">
6146  */
6147
6148 /**
6149  * @class Roo.EventManager
6150  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6151  * several useful events directly.
6152  * See {@link Roo.EventObject} for more details on normalized event objects.
6153  * @singleton
6154  */
6155 Roo.EventManager = function(){
6156     var docReadyEvent, docReadyProcId, docReadyState = false;
6157     var resizeEvent, resizeTask, textEvent, textSize;
6158     var E = Roo.lib.Event;
6159     var D = Roo.lib.Dom;
6160
6161     
6162     
6163
6164     var fireDocReady = function(){
6165         if(!docReadyState){
6166             docReadyState = true;
6167             Roo.isReady = true;
6168             if(docReadyProcId){
6169                 clearInterval(docReadyProcId);
6170             }
6171             if(Roo.isGecko || Roo.isOpera) {
6172                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6173             }
6174             if(Roo.isIE){
6175                 var defer = document.getElementById("ie-deferred-loader");
6176                 if(defer){
6177                     defer.onreadystatechange = null;
6178                     defer.parentNode.removeChild(defer);
6179                 }
6180             }
6181             if(docReadyEvent){
6182                 docReadyEvent.fire();
6183                 docReadyEvent.clearListeners();
6184             }
6185         }
6186     };
6187     
6188     var initDocReady = function(){
6189         docReadyEvent = new Roo.util.Event();
6190         if(Roo.isGecko || Roo.isOpera) {
6191             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6192         }else if(Roo.isIE){
6193             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6194             var defer = document.getElementById("ie-deferred-loader");
6195             defer.onreadystatechange = function(){
6196                 if(this.readyState == "complete"){
6197                     fireDocReady();
6198                 }
6199             };
6200         }else if(Roo.isSafari){ 
6201             docReadyProcId = setInterval(function(){
6202                 var rs = document.readyState;
6203                 if(rs == "complete") {
6204                     fireDocReady();     
6205                  }
6206             }, 10);
6207         }
6208         // no matter what, make sure it fires on load
6209         E.on(window, "load", fireDocReady);
6210     };
6211
6212     var createBuffered = function(h, o){
6213         var task = new Roo.util.DelayedTask(h);
6214         return function(e){
6215             // create new event object impl so new events don't wipe out properties
6216             e = new Roo.EventObjectImpl(e);
6217             task.delay(o.buffer, h, null, [e]);
6218         };
6219     };
6220
6221     var createSingle = function(h, el, ename, fn){
6222         return function(e){
6223             Roo.EventManager.removeListener(el, ename, fn);
6224             h(e);
6225         };
6226     };
6227
6228     var createDelayed = function(h, o){
6229         return function(e){
6230             // create new event object impl so new events don't wipe out properties
6231             e = new Roo.EventObjectImpl(e);
6232             setTimeout(function(){
6233                 h(e);
6234             }, o.delay || 10);
6235         };
6236     };
6237     var transitionEndVal = false;
6238     
6239     var transitionEnd = function()
6240     {
6241         if (transitionEndVal) {
6242             return transitionEndVal;
6243         }
6244         var el = document.createElement('div');
6245
6246         var transEndEventNames = {
6247             WebkitTransition : 'webkitTransitionEnd',
6248             MozTransition    : 'transitionend',
6249             OTransition      : 'oTransitionEnd otransitionend',
6250             transition       : 'transitionend'
6251         };
6252     
6253         for (var name in transEndEventNames) {
6254             if (el.style[name] !== undefined) {
6255                 transitionEndVal = transEndEventNames[name];
6256                 return  transitionEndVal ;
6257             }
6258         }
6259     }
6260     
6261
6262     var listen = function(element, ename, opt, fn, scope){
6263         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6264         fn = fn || o.fn; scope = scope || o.scope;
6265         var el = Roo.getDom(element);
6266         
6267         
6268         if(!el){
6269             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6270         }
6271         
6272         if (ename == 'transitionend') {
6273             ename = transitionEnd();
6274         }
6275         var h = function(e){
6276             e = Roo.EventObject.setEvent(e);
6277             var t;
6278             if(o.delegate){
6279                 t = e.getTarget(o.delegate, el);
6280                 if(!t){
6281                     return;
6282                 }
6283             }else{
6284                 t = e.target;
6285             }
6286             if(o.stopEvent === true){
6287                 e.stopEvent();
6288             }
6289             if(o.preventDefault === true){
6290                e.preventDefault();
6291             }
6292             if(o.stopPropagation === true){
6293                 e.stopPropagation();
6294             }
6295
6296             if(o.normalized === false){
6297                 e = e.browserEvent;
6298             }
6299
6300             fn.call(scope || el, e, t, o);
6301         };
6302         if(o.delay){
6303             h = createDelayed(h, o);
6304         }
6305         if(o.single){
6306             h = createSingle(h, el, ename, fn);
6307         }
6308         if(o.buffer){
6309             h = createBuffered(h, o);
6310         }
6311         
6312         fn._handlers = fn._handlers || [];
6313         
6314         
6315         fn._handlers.push([Roo.id(el), ename, h]);
6316         
6317         
6318          
6319         E.on(el, ename, h);
6320         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6321             el.addEventListener("DOMMouseScroll", h, false);
6322             E.on(window, 'unload', function(){
6323                 el.removeEventListener("DOMMouseScroll", h, false);
6324             });
6325         }
6326         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6327             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6328         }
6329         return h;
6330     };
6331
6332     var stopListening = function(el, ename, fn){
6333         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6334         if(hds){
6335             for(var i = 0, len = hds.length; i < len; i++){
6336                 var h = hds[i];
6337                 if(h[0] == id && h[1] == ename){
6338                     hd = h[2];
6339                     hds.splice(i, 1);
6340                     break;
6341                 }
6342             }
6343         }
6344         E.un(el, ename, hd);
6345         el = Roo.getDom(el);
6346         if(ename == "mousewheel" && el.addEventListener){
6347             el.removeEventListener("DOMMouseScroll", hd, false);
6348         }
6349         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6350             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6351         }
6352     };
6353
6354     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6355     
6356     var pub = {
6357         
6358         
6359         /** 
6360          * Fix for doc tools
6361          * @scope Roo.EventManager
6362          */
6363         
6364         
6365         /** 
6366          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6367          * object with a Roo.EventObject
6368          * @param {Function} fn        The method the event invokes
6369          * @param {Object}   scope    An object that becomes the scope of the handler
6370          * @param {boolean}  override If true, the obj passed in becomes
6371          *                             the execution scope of the listener
6372          * @return {Function} The wrapped function
6373          * @deprecated
6374          */
6375         wrap : function(fn, scope, override){
6376             return function(e){
6377                 Roo.EventObject.setEvent(e);
6378                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6379             };
6380         },
6381         
6382         /**
6383      * Appends an event handler to an element (shorthand for addListener)
6384      * @param {String/HTMLElement}   element        The html element or id to assign the
6385      * @param {String}   eventName The type of event to listen for
6386      * @param {Function} handler The method the event invokes
6387      * @param {Object}   scope (optional) The scope in which to execute the handler
6388      * function. The handler function's "this" context.
6389      * @param {Object}   options (optional) An object containing handler configuration
6390      * properties. This may contain any of the following properties:<ul>
6391      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6392      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6393      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6394      * <li>preventDefault {Boolean} True to prevent the default action</li>
6395      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6396      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6397      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6398      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6399      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6400      * by the specified number of milliseconds. If the event fires again within that time, the original
6401      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6402      * </ul><br>
6403      * <p>
6404      * <b>Combining Options</b><br>
6405      * Using the options argument, it is possible to combine different types of listeners:<br>
6406      * <br>
6407      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6408      * Code:<pre><code>
6409 el.on('click', this.onClick, this, {
6410     single: true,
6411     delay: 100,
6412     stopEvent : true,
6413     forumId: 4
6414 });</code></pre>
6415      * <p>
6416      * <b>Attaching multiple handlers in 1 call</b><br>
6417       * The method also allows for a single argument to be passed which is a config object containing properties
6418      * which specify multiple handlers.
6419      * <p>
6420      * Code:<pre><code>
6421 el.on({
6422     'click' : {
6423         fn: this.onClick
6424         scope: this,
6425         delay: 100
6426     },
6427     'mouseover' : {
6428         fn: this.onMouseOver
6429         scope: this
6430     },
6431     'mouseout' : {
6432         fn: this.onMouseOut
6433         scope: this
6434     }
6435 });</code></pre>
6436      * <p>
6437      * Or a shorthand syntax:<br>
6438      * Code:<pre><code>
6439 el.on({
6440     'click' : this.onClick,
6441     'mouseover' : this.onMouseOver,
6442     'mouseout' : this.onMouseOut
6443     scope: this
6444 });</code></pre>
6445      */
6446         addListener : function(element, eventName, fn, scope, options){
6447             if(typeof eventName == "object"){
6448                 var o = eventName;
6449                 for(var e in o){
6450                     if(propRe.test(e)){
6451                         continue;
6452                     }
6453                     if(typeof o[e] == "function"){
6454                         // shared options
6455                         listen(element, e, o, o[e], o.scope);
6456                     }else{
6457                         // individual options
6458                         listen(element, e, o[e]);
6459                     }
6460                 }
6461                 return;
6462             }
6463             return listen(element, eventName, options, fn, scope);
6464         },
6465         
6466         /**
6467          * Removes an event handler
6468          *
6469          * @param {String/HTMLElement}   element        The id or html element to remove the 
6470          *                             event from
6471          * @param {String}   eventName     The type of event
6472          * @param {Function} fn
6473          * @return {Boolean} True if a listener was actually removed
6474          */
6475         removeListener : function(element, eventName, fn){
6476             return stopListening(element, eventName, fn);
6477         },
6478         
6479         /**
6480          * Fires when the document is ready (before onload and before images are loaded). Can be 
6481          * accessed shorthanded Roo.onReady().
6482          * @param {Function} fn        The method the event invokes
6483          * @param {Object}   scope    An  object that becomes the scope of the handler
6484          * @param {boolean}  options
6485          */
6486         onDocumentReady : function(fn, scope, options){
6487             if(docReadyState){ // if it already fired
6488                 docReadyEvent.addListener(fn, scope, options);
6489                 docReadyEvent.fire();
6490                 docReadyEvent.clearListeners();
6491                 return;
6492             }
6493             if(!docReadyEvent){
6494                 initDocReady();
6495             }
6496             docReadyEvent.addListener(fn, scope, options);
6497         },
6498         
6499         /**
6500          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6501          * @param {Function} fn        The method the event invokes
6502          * @param {Object}   scope    An object that becomes the scope of the handler
6503          * @param {boolean}  options
6504          */
6505         onWindowResize : function(fn, scope, options){
6506             if(!resizeEvent){
6507                 resizeEvent = new Roo.util.Event();
6508                 resizeTask = new Roo.util.DelayedTask(function(){
6509                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6510                 });
6511                 E.on(window, "resize", function(){
6512                     if(Roo.isIE){
6513                         resizeTask.delay(50);
6514                     }else{
6515                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6516                     }
6517                 });
6518             }
6519             resizeEvent.addListener(fn, scope, options);
6520         },
6521
6522         /**
6523          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6524          * @param {Function} fn        The method the event invokes
6525          * @param {Object}   scope    An object that becomes the scope of the handler
6526          * @param {boolean}  options
6527          */
6528         onTextResize : function(fn, scope, options){
6529             if(!textEvent){
6530                 textEvent = new Roo.util.Event();
6531                 var textEl = new Roo.Element(document.createElement('div'));
6532                 textEl.dom.className = 'x-text-resize';
6533                 textEl.dom.innerHTML = 'X';
6534                 textEl.appendTo(document.body);
6535                 textSize = textEl.dom.offsetHeight;
6536                 setInterval(function(){
6537                     if(textEl.dom.offsetHeight != textSize){
6538                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6539                     }
6540                 }, this.textResizeInterval);
6541             }
6542             textEvent.addListener(fn, scope, options);
6543         },
6544
6545         /**
6546          * Removes the passed window resize listener.
6547          * @param {Function} fn        The method the event invokes
6548          * @param {Object}   scope    The scope of handler
6549          */
6550         removeResizeListener : function(fn, scope){
6551             if(resizeEvent){
6552                 resizeEvent.removeListener(fn, scope);
6553             }
6554         },
6555
6556         // private
6557         fireResize : function(){
6558             if(resizeEvent){
6559                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6560             }   
6561         },
6562         /**
6563          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6564          */
6565         ieDeferSrc : false,
6566         /**
6567          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6568          */
6569         textResizeInterval : 50
6570     };
6571     
6572     /**
6573      * Fix for doc tools
6574      * @scopeAlias pub=Roo.EventManager
6575      */
6576     
6577      /**
6578      * Appends an event handler to an element (shorthand for addListener)
6579      * @param {String/HTMLElement}   element        The html element or id to assign the
6580      * @param {String}   eventName The type of event to listen for
6581      * @param {Function} handler The method the event invokes
6582      * @param {Object}   scope (optional) The scope in which to execute the handler
6583      * function. The handler function's "this" context.
6584      * @param {Object}   options (optional) An object containing handler configuration
6585      * properties. This may contain any of the following properties:<ul>
6586      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6587      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6588      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6589      * <li>preventDefault {Boolean} True to prevent the default action</li>
6590      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6591      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6592      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6593      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6594      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6595      * by the specified number of milliseconds. If the event fires again within that time, the original
6596      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6597      * </ul><br>
6598      * <p>
6599      * <b>Combining Options</b><br>
6600      * Using the options argument, it is possible to combine different types of listeners:<br>
6601      * <br>
6602      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6603      * Code:<pre><code>
6604 el.on('click', this.onClick, this, {
6605     single: true,
6606     delay: 100,
6607     stopEvent : true,
6608     forumId: 4
6609 });</code></pre>
6610      * <p>
6611      * <b>Attaching multiple handlers in 1 call</b><br>
6612       * The method also allows for a single argument to be passed which is a config object containing properties
6613      * which specify multiple handlers.
6614      * <p>
6615      * Code:<pre><code>
6616 el.on({
6617     'click' : {
6618         fn: this.onClick
6619         scope: this,
6620         delay: 100
6621     },
6622     'mouseover' : {
6623         fn: this.onMouseOver
6624         scope: this
6625     },
6626     'mouseout' : {
6627         fn: this.onMouseOut
6628         scope: this
6629     }
6630 });</code></pre>
6631      * <p>
6632      * Or a shorthand syntax:<br>
6633      * Code:<pre><code>
6634 el.on({
6635     'click' : this.onClick,
6636     'mouseover' : this.onMouseOver,
6637     'mouseout' : this.onMouseOut
6638     scope: this
6639 });</code></pre>
6640      */
6641     pub.on = pub.addListener;
6642     pub.un = pub.removeListener;
6643
6644     pub.stoppedMouseDownEvent = new Roo.util.Event();
6645     return pub;
6646 }();
6647 /**
6648   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6649   * @param {Function} fn        The method the event invokes
6650   * @param {Object}   scope    An  object that becomes the scope of the handler
6651   * @param {boolean}  override If true, the obj passed in becomes
6652   *                             the execution scope of the listener
6653   * @member Roo
6654   * @method onReady
6655  */
6656 Roo.onReady = Roo.EventManager.onDocumentReady;
6657
6658 Roo.onReady(function(){
6659     var bd = Roo.get(document.body);
6660     if(!bd){ return; }
6661
6662     var cls = [
6663             Roo.isIE ? "roo-ie"
6664             : Roo.isIE11 ? "roo-ie11"
6665             : Roo.isEdge ? "roo-edge"
6666             : Roo.isGecko ? "roo-gecko"
6667             : Roo.isOpera ? "roo-opera"
6668             : Roo.isSafari ? "roo-safari" : ""];
6669
6670     if(Roo.isMac){
6671         cls.push("roo-mac");
6672     }
6673     if(Roo.isLinux){
6674         cls.push("roo-linux");
6675     }
6676     if(Roo.isIOS){
6677         cls.push("roo-ios");
6678     }
6679     if(Roo.isTouch){
6680         cls.push("roo-touch");
6681     }
6682     if(Roo.isBorderBox){
6683         cls.push('roo-border-box');
6684     }
6685     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6686         var p = bd.dom.parentNode;
6687         if(p){
6688             p.className += ' roo-strict';
6689         }
6690     }
6691     bd.addClass(cls.join(' '));
6692 });
6693
6694 /**
6695  * @class Roo.EventObject
6696  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6697  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6698  * Example:
6699  * <pre><code>
6700  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6701     e.preventDefault();
6702     var target = e.getTarget();
6703     ...
6704  }
6705  var myDiv = Roo.get("myDiv");
6706  myDiv.on("click", handleClick);
6707  //or
6708  Roo.EventManager.on("myDiv", 'click', handleClick);
6709  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6710  </code></pre>
6711  * @singleton
6712  */
6713 Roo.EventObject = function(){
6714     
6715     var E = Roo.lib.Event;
6716     
6717     // safari keypress events for special keys return bad keycodes
6718     var safariKeys = {
6719         63234 : 37, // left
6720         63235 : 39, // right
6721         63232 : 38, // up
6722         63233 : 40, // down
6723         63276 : 33, // page up
6724         63277 : 34, // page down
6725         63272 : 46, // delete
6726         63273 : 36, // home
6727         63275 : 35  // end
6728     };
6729
6730     // normalize button clicks
6731     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6732                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6733
6734     Roo.EventObjectImpl = function(e){
6735         if(e){
6736             this.setEvent(e.browserEvent || e);
6737         }
6738     };
6739     Roo.EventObjectImpl.prototype = {
6740         /**
6741          * Used to fix doc tools.
6742          * @scope Roo.EventObject.prototype
6743          */
6744             
6745
6746         
6747         
6748         /** The normal browser event */
6749         browserEvent : null,
6750         /** The button pressed in a mouse event */
6751         button : -1,
6752         /** True if the shift key was down during the event */
6753         shiftKey : false,
6754         /** True if the control key was down during the event */
6755         ctrlKey : false,
6756         /** True if the alt key was down during the event */
6757         altKey : false,
6758
6759         /** Key constant 
6760         * @type Number */
6761         BACKSPACE : 8,
6762         /** Key constant 
6763         * @type Number */
6764         TAB : 9,
6765         /** Key constant 
6766         * @type Number */
6767         RETURN : 13,
6768         /** Key constant 
6769         * @type Number */
6770         ENTER : 13,
6771         /** Key constant 
6772         * @type Number */
6773         SHIFT : 16,
6774         /** Key constant 
6775         * @type Number */
6776         CONTROL : 17,
6777         /** Key constant 
6778         * @type Number */
6779         ESC : 27,
6780         /** Key constant 
6781         * @type Number */
6782         SPACE : 32,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEUP : 33,
6786         /** Key constant 
6787         * @type Number */
6788         PAGEDOWN : 34,
6789         /** Key constant 
6790         * @type Number */
6791         END : 35,
6792         /** Key constant 
6793         * @type Number */
6794         HOME : 36,
6795         /** Key constant 
6796         * @type Number */
6797         LEFT : 37,
6798         /** Key constant 
6799         * @type Number */
6800         UP : 38,
6801         /** Key constant 
6802         * @type Number */
6803         RIGHT : 39,
6804         /** Key constant 
6805         * @type Number */
6806         DOWN : 40,
6807         /** Key constant 
6808         * @type Number */
6809         DELETE : 46,
6810         /** Key constant 
6811         * @type Number */
6812         F5 : 116,
6813
6814            /** @private */
6815         setEvent : function(e){
6816             if(e == this || (e && e.browserEvent)){ // already wrapped
6817                 return e;
6818             }
6819             this.browserEvent = e;
6820             if(e){
6821                 // normalize buttons
6822                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6823                 if(e.type == 'click' && this.button == -1){
6824                     this.button = 0;
6825                 }
6826                 this.type = e.type;
6827                 this.shiftKey = e.shiftKey;
6828                 // mac metaKey behaves like ctrlKey
6829                 this.ctrlKey = e.ctrlKey || e.metaKey;
6830                 this.altKey = e.altKey;
6831                 // in getKey these will be normalized for the mac
6832                 this.keyCode = e.keyCode;
6833                 // keyup warnings on firefox.
6834                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6835                 // cache the target for the delayed and or buffered events
6836                 this.target = E.getTarget(e);
6837                 // same for XY
6838                 this.xy = E.getXY(e);
6839             }else{
6840                 this.button = -1;
6841                 this.shiftKey = false;
6842                 this.ctrlKey = false;
6843                 this.altKey = false;
6844                 this.keyCode = 0;
6845                 this.charCode =0;
6846                 this.target = null;
6847                 this.xy = [0, 0];
6848             }
6849             return this;
6850         },
6851
6852         /**
6853          * Stop the event (preventDefault and stopPropagation)
6854          */
6855         stopEvent : function(){
6856             if(this.browserEvent){
6857                 if(this.browserEvent.type == 'mousedown'){
6858                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6859                 }
6860                 E.stopEvent(this.browserEvent);
6861             }
6862         },
6863
6864         /**
6865          * Prevents the browsers default handling of the event.
6866          */
6867         preventDefault : function(){
6868             if(this.browserEvent){
6869                 E.preventDefault(this.browserEvent);
6870             }
6871         },
6872
6873         /** @private */
6874         isNavKeyPress : function(){
6875             var k = this.keyCode;
6876             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6877             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6878         },
6879
6880         isSpecialKey : function(){
6881             var k = this.keyCode;
6882             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6883             (k == 16) || (k == 17) ||
6884             (k >= 18 && k <= 20) ||
6885             (k >= 33 && k <= 35) ||
6886             (k >= 36 && k <= 39) ||
6887             (k >= 44 && k <= 45);
6888         },
6889         /**
6890          * Cancels bubbling of the event.
6891          */
6892         stopPropagation : function(){
6893             if(this.browserEvent){
6894                 if(this.type == 'mousedown'){
6895                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6896                 }
6897                 E.stopPropagation(this.browserEvent);
6898             }
6899         },
6900
6901         /**
6902          * Gets the key code for the event.
6903          * @return {Number}
6904          */
6905         getCharCode : function(){
6906             return this.charCode || this.keyCode;
6907         },
6908
6909         /**
6910          * Returns a normalized keyCode for the event.
6911          * @return {Number} The key code
6912          */
6913         getKey : function(){
6914             var k = this.keyCode || this.charCode;
6915             return Roo.isSafari ? (safariKeys[k] || k) : k;
6916         },
6917
6918         /**
6919          * Gets the x coordinate of the event.
6920          * @return {Number}
6921          */
6922         getPageX : function(){
6923             return this.xy[0];
6924         },
6925
6926         /**
6927          * Gets the y coordinate of the event.
6928          * @return {Number}
6929          */
6930         getPageY : function(){
6931             return this.xy[1];
6932         },
6933
6934         /**
6935          * Gets the time of the event.
6936          * @return {Number}
6937          */
6938         getTime : function(){
6939             if(this.browserEvent){
6940                 return E.getTime(this.browserEvent);
6941             }
6942             return null;
6943         },
6944
6945         /**
6946          * Gets the page coordinates of the event.
6947          * @return {Array} The xy values like [x, y]
6948          */
6949         getXY : function(){
6950             return this.xy;
6951         },
6952
6953         /**
6954          * Gets the target for the event.
6955          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6956          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6957                 search as a number or element (defaults to 10 || document.body)
6958          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6959          * @return {HTMLelement}
6960          */
6961         getTarget : function(selector, maxDepth, returnEl){
6962             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6963         },
6964         /**
6965          * Gets the related target.
6966          * @return {HTMLElement}
6967          */
6968         getRelatedTarget : function(){
6969             if(this.browserEvent){
6970                 return E.getRelatedTarget(this.browserEvent);
6971             }
6972             return null;
6973         },
6974
6975         /**
6976          * Normalizes mouse wheel delta across browsers
6977          * @return {Number} The delta
6978          */
6979         getWheelDelta : function(){
6980             var e = this.browserEvent;
6981             var delta = 0;
6982             if(e.wheelDelta){ /* IE/Opera. */
6983                 delta = e.wheelDelta/120;
6984             }else if(e.detail){ /* Mozilla case. */
6985                 delta = -e.detail/3;
6986             }
6987             return delta;
6988         },
6989
6990         /**
6991          * Returns true if the control, meta, shift or alt key was pressed during this event.
6992          * @return {Boolean}
6993          */
6994         hasModifier : function(){
6995             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6996         },
6997
6998         /**
6999          * Returns true if the target of this event equals el or is a child of el
7000          * @param {String/HTMLElement/Element} el
7001          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
7002          * @return {Boolean}
7003          */
7004         within : function(el, related){
7005             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7006             return t && Roo.fly(el).contains(t);
7007         },
7008
7009         getPoint : function(){
7010             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7011         }
7012     };
7013
7014     return new Roo.EventObjectImpl();
7015 }();
7016             
7017     /*
7018  * Based on:
7019  * Ext JS Library 1.1.1
7020  * Copyright(c) 2006-2007, Ext JS, LLC.
7021  *
7022  * Originally Released Under LGPL - original licence link has changed is not relivant.
7023  *
7024  * Fork - LGPL
7025  * <script type="text/javascript">
7026  */
7027
7028  
7029 // was in Composite Element!??!?!
7030  
7031 (function(){
7032     var D = Roo.lib.Dom;
7033     var E = Roo.lib.Event;
7034     var A = Roo.lib.Anim;
7035
7036     // local style camelizing for speed
7037     var propCache = {};
7038     var camelRe = /(-[a-z])/gi;
7039     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7040     var view = document.defaultView;
7041
7042 /**
7043  * @class Roo.Element
7044  * Represents an Element in the DOM.<br><br>
7045  * Usage:<br>
7046 <pre><code>
7047 var el = Roo.get("my-div");
7048
7049 // or with getEl
7050 var el = getEl("my-div");
7051
7052 // or with a DOM element
7053 var el = Roo.get(myDivElement);
7054 </code></pre>
7055  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7056  * each call instead of constructing a new one.<br><br>
7057  * <b>Animations</b><br />
7058  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7059  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7060 <pre>
7061 Option    Default   Description
7062 --------- --------  ---------------------------------------------
7063 duration  .35       The duration of the animation in seconds
7064 easing    easeOut   The YUI easing method
7065 callback  none      A function to execute when the anim completes
7066 scope     this      The scope (this) of the callback function
7067 </pre>
7068 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7069 * manipulate the animation. Here's an example:
7070 <pre><code>
7071 var el = Roo.get("my-div");
7072
7073 // no animation
7074 el.setWidth(100);
7075
7076 // default animation
7077 el.setWidth(100, true);
7078
7079 // animation with some options set
7080 el.setWidth(100, {
7081     duration: 1,
7082     callback: this.foo,
7083     scope: this
7084 });
7085
7086 // using the "anim" property to get the Anim object
7087 var opt = {
7088     duration: 1,
7089     callback: this.foo,
7090     scope: this
7091 };
7092 el.setWidth(100, opt);
7093 ...
7094 if(opt.anim.isAnimated()){
7095     opt.anim.stop();
7096 }
7097 </code></pre>
7098 * <b> Composite (Collections of) Elements</b><br />
7099  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7100  * @constructor Create a new Element directly.
7101  * @param {String/HTMLElement} element
7102  * @param {Boolean} forceNew (optional) By default the constructor checks to see if there is already an instance of this element in the cache and if there is it returns the same instance. This will skip that check (useful for extending this class).
7103  */
7104     Roo.Element = function(element, forceNew){
7105         var dom = typeof element == "string" ?
7106                 document.getElementById(element) : element;
7107         if(!dom){ // invalid id/element
7108             return null;
7109         }
7110         var id = dom.id;
7111         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7112             return Roo.Element.cache[id];
7113         }
7114
7115         /**
7116          * The DOM element
7117          * @type HTMLElement
7118          */
7119         this.dom = dom;
7120
7121         /**
7122          * The DOM element ID
7123          * @type String
7124          */
7125         this.id = id || Roo.id(dom);
7126     };
7127
7128     var El = Roo.Element;
7129
7130     El.prototype = {
7131         /**
7132          * The element's default display mode  (defaults to "")
7133          * @type String
7134          */
7135         originalDisplay : "",
7136
7137         visibilityMode : 1,
7138         /**
7139          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7140          * @type String
7141          */
7142         defaultUnit : "px",
7143         
7144         /**
7145          * Sets the element's visibility mode. When setVisible() is called it
7146          * will use this to determine whether to set the visibility or the display property.
7147          * @param visMode Element.VISIBILITY or Element.DISPLAY
7148          * @return {Roo.Element} this
7149          */
7150         setVisibilityMode : function(visMode){
7151             this.visibilityMode = visMode;
7152             return this;
7153         },
7154         /**
7155          * Convenience method for setVisibilityMode(Element.DISPLAY)
7156          * @param {String} display (optional) What to set display to when visible
7157          * @return {Roo.Element} this
7158          */
7159         enableDisplayMode : function(display){
7160             this.setVisibilityMode(El.DISPLAY);
7161             if(typeof display != "undefined") { this.originalDisplay = display; }
7162             return this;
7163         },
7164
7165         /**
7166          * Looks at this node and then at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7167          * @param {String} selector The simple selector to test
7168          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7169                 search as a number or element (defaults to 10 || document.body)
7170          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7171          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7172          */
7173         findParent : function(simpleSelector, maxDepth, returnEl){
7174             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7175             maxDepth = maxDepth || 50;
7176             if(typeof maxDepth != "number"){
7177                 stopEl = Roo.getDom(maxDepth);
7178                 maxDepth = 10;
7179             }
7180             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7181                 if(dq.is(p, simpleSelector)){
7182                     return returnEl ? Roo.get(p) : p;
7183                 }
7184                 depth++;
7185                 p = p.parentNode;
7186             }
7187             return null;
7188         },
7189
7190
7191         /**
7192          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7193          * @param {String} selector The simple selector to test
7194          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7195                 search as a number or element (defaults to 10 || document.body)
7196          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7197          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7198          */
7199         findParentNode : function(simpleSelector, maxDepth, returnEl){
7200             var p = Roo.fly(this.dom.parentNode, '_internal');
7201             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7202         },
7203         
7204         /**
7205          * Looks at  the scrollable parent element
7206          */
7207         findScrollableParent : function()
7208         {
7209             var overflowRegex = /(auto|scroll)/;
7210             
7211             if(this.getStyle('position') === 'fixed'){
7212                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7213             }
7214             
7215             var excludeStaticParent = this.getStyle('position') === "absolute";
7216             
7217             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7218                 
7219                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7220                     continue;
7221                 }
7222                 
7223                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7224                     return parent;
7225                 }
7226                 
7227                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7228                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7229                 }
7230             }
7231             
7232             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7233         },
7234
7235         /**
7236          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7237          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7238          * @param {String} selector The simple selector to test
7239          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7240                 search as a number or element (defaults to 10 || document.body)
7241          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7242          */
7243         up : function(simpleSelector, maxDepth){
7244             return this.findParentNode(simpleSelector, maxDepth, true);
7245         },
7246
7247
7248
7249         /**
7250          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7251          * @param {String} selector The simple selector to test
7252          * @return {Boolean} True if this element matches the selector, else false
7253          */
7254         is : function(simpleSelector){
7255             return Roo.DomQuery.is(this.dom, simpleSelector);
7256         },
7257
7258         /**
7259          * Perform animation on this element.
7260          * @param {Object} args The YUI animation control args
7261          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7262          * @param {Function} onComplete (optional) Function to call when animation completes
7263          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7264          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7265          * @return {Roo.Element} this
7266          */
7267         animate : function(args, duration, onComplete, easing, animType){
7268             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7269             return this;
7270         },
7271
7272         /*
7273          * @private Internal animation call
7274          */
7275         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7276             animType = animType || 'run';
7277             opt = opt || {};
7278             var anim = Roo.lib.Anim[animType](
7279                 this.dom, args,
7280                 (opt.duration || defaultDur) || .35,
7281                 (opt.easing || defaultEase) || 'easeOut',
7282                 function(){
7283                     Roo.callback(cb, this);
7284                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7285                 },
7286                 this
7287             );
7288             opt.anim = anim;
7289             return anim;
7290         },
7291
7292         // private legacy anim prep
7293         preanim : function(a, i){
7294             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7295         },
7296
7297         /**
7298          * Removes worthless text nodes
7299          * @param {Boolean} forceReclean (optional) By default the element
7300          * keeps track if it has been cleaned already so
7301          * you can call this over and over. However, if you update the element and
7302          * need to force a reclean, you can pass true.
7303          */
7304         clean : function(forceReclean){
7305             if(this.isCleaned && forceReclean !== true){
7306                 return this;
7307             }
7308             var ns = /\S/;
7309             var d = this.dom, n = d.firstChild, ni = -1;
7310             while(n){
7311                 var nx = n.nextSibling;
7312                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7313                     d.removeChild(n);
7314                 }else{
7315                     n.nodeIndex = ++ni;
7316                 }
7317                 n = nx;
7318             }
7319             this.isCleaned = true;
7320             return this;
7321         },
7322
7323         // private
7324         calcOffsetsTo : function(el){
7325             el = Roo.get(el);
7326             var d = el.dom;
7327             var restorePos = false;
7328             if(el.getStyle('position') == 'static'){
7329                 el.position('relative');
7330                 restorePos = true;
7331             }
7332             var x = 0, y =0;
7333             var op = this.dom;
7334             while(op && op != d && op.tagName != 'HTML'){
7335                 x+= op.offsetLeft;
7336                 y+= op.offsetTop;
7337                 op = op.offsetParent;
7338             }
7339             if(restorePos){
7340                 el.position('static');
7341             }
7342             return [x, y];
7343         },
7344
7345         /**
7346          * Scrolls this element into view within the passed container.
7347          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7348          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7349          * @return {Roo.Element} this
7350          */
7351         scrollIntoView : function(container, hscroll){
7352             var c = Roo.getDom(container) || document.body;
7353             var el = this.dom;
7354
7355             var o = this.calcOffsetsTo(c),
7356                 l = o[0],
7357                 t = o[1],
7358                 b = t+el.offsetHeight,
7359                 r = l+el.offsetWidth;
7360
7361             var ch = c.clientHeight;
7362             var ct = parseInt(c.scrollTop, 10);
7363             var cl = parseInt(c.scrollLeft, 10);
7364             var cb = ct + ch;
7365             var cr = cl + c.clientWidth;
7366
7367             if(t < ct){
7368                 c.scrollTop = t;
7369             }else if(b > cb){
7370                 c.scrollTop = b-ch;
7371             }
7372
7373             if(hscroll !== false){
7374                 if(l < cl){
7375                     c.scrollLeft = l;
7376                 }else if(r > cr){
7377                     c.scrollLeft = r-c.clientWidth;
7378                 }
7379             }
7380             return this;
7381         },
7382
7383         // private
7384         scrollChildIntoView : function(child, hscroll){
7385             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7386         },
7387
7388         /**
7389          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7390          * the new height may not be available immediately.
7391          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7392          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7393          * @param {Function} onComplete (optional) Function to call when animation completes
7394          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7395          * @return {Roo.Element} this
7396          */
7397         autoHeight : function(animate, duration, onComplete, easing){
7398             var oldHeight = this.getHeight();
7399             this.clip();
7400             this.setHeight(1); // force clipping
7401             setTimeout(function(){
7402                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7403                 if(!animate){
7404                     this.setHeight(height);
7405                     this.unclip();
7406                     if(typeof onComplete == "function"){
7407                         onComplete();
7408                     }
7409                 }else{
7410                     this.setHeight(oldHeight); // restore original height
7411                     this.setHeight(height, animate, duration, function(){
7412                         this.unclip();
7413                         if(typeof onComplete == "function") { onComplete(); }
7414                     }.createDelegate(this), easing);
7415                 }
7416             }.createDelegate(this), 0);
7417             return this;
7418         },
7419
7420         /**
7421          * Returns true if this element is an ancestor of the passed element
7422          * @param {HTMLElement/String} el The element to check
7423          * @return {Boolean} True if this element is an ancestor of el, else false
7424          */
7425         contains : function(el){
7426             if(!el){return false;}
7427             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7428         },
7429
7430         /**
7431          * Checks whether the element is currently visible using both visibility and display properties.
7432          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7433          * @return {Boolean} True if the element is currently visible, else false
7434          */
7435         isVisible : function(deep) {
7436             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7437             if(deep !== true || !vis){
7438                 return vis;
7439             }
7440             var p = this.dom.parentNode;
7441             while(p && p.tagName.toLowerCase() != "body"){
7442                 if(!Roo.fly(p, '_isVisible').isVisible()){
7443                     return false;
7444                 }
7445                 p = p.parentNode;
7446             }
7447             return true;
7448         },
7449
7450         /**
7451          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7452          * @param {String} selector The CSS selector
7453          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7454          * @return {CompositeElement/CompositeElementLite} The composite element
7455          */
7456         select : function(selector, unique){
7457             return El.select(selector, unique, this.dom);
7458         },
7459
7460         /**
7461          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7462          * @param {String} selector The CSS selector
7463          * @return {Array} An array of the matched nodes
7464          */
7465         query : function(selector, unique){
7466             return Roo.DomQuery.select(selector, this.dom);
7467         },
7468
7469         /**
7470          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7471          * @param {String} selector The CSS selector
7472          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7473          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7474          */
7475         child : function(selector, returnDom){
7476             var n = Roo.DomQuery.selectNode(selector, this.dom);
7477             return returnDom ? n : Roo.get(n);
7478         },
7479
7480         /**
7481          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7482          * @param {String} selector The CSS selector
7483          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7484          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7485          */
7486         down : function(selector, returnDom){
7487             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7488             return returnDom ? n : Roo.get(n);
7489         },
7490
7491         /**
7492          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7493          * @param {String} group The group the DD object is member of
7494          * @param {Object} config The DD config object
7495          * @param {Object} overrides An object containing methods to override/implement on the DD object
7496          * @return {Roo.dd.DD} The DD object
7497          */
7498         initDD : function(group, config, overrides){
7499             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7500             return Roo.apply(dd, overrides);
7501         },
7502
7503         /**
7504          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7505          * @param {String} group The group the DDProxy object is member of
7506          * @param {Object} config The DDProxy config object
7507          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7508          * @return {Roo.dd.DDProxy} The DDProxy object
7509          */
7510         initDDProxy : function(group, config, overrides){
7511             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7512             return Roo.apply(dd, overrides);
7513         },
7514
7515         /**
7516          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7517          * @param {String} group The group the DDTarget object is member of
7518          * @param {Object} config The DDTarget config object
7519          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7520          * @return {Roo.dd.DDTarget} The DDTarget object
7521          */
7522         initDDTarget : function(group, config, overrides){
7523             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7524             return Roo.apply(dd, overrides);
7525         },
7526
7527         /**
7528          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7529          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7530          * @param {Boolean} visible Whether the element is visible
7531          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7532          * @return {Roo.Element} this
7533          */
7534          setVisible : function(visible, animate){
7535             if(!animate || !A){
7536                 if(this.visibilityMode == El.DISPLAY){
7537                     this.setDisplayed(visible);
7538                 }else{
7539                     this.fixDisplay();
7540                     this.dom.style.visibility = visible ? "visible" : "hidden";
7541                 }
7542             }else{
7543                 // closure for composites
7544                 var dom = this.dom;
7545                 var visMode = this.visibilityMode;
7546                 if(visible){
7547                     this.setOpacity(.01);
7548                     this.setVisible(true);
7549                 }
7550                 this.anim({opacity: { to: (visible?1:0) }},
7551                       this.preanim(arguments, 1),
7552                       null, .35, 'easeIn', function(){
7553                          if(!visible){
7554                              if(visMode == El.DISPLAY){
7555                                  dom.style.display = "none";
7556                              }else{
7557                                  dom.style.visibility = "hidden";
7558                              }
7559                              Roo.get(dom).setOpacity(1);
7560                          }
7561                      });
7562             }
7563             return this;
7564         },
7565
7566         /**
7567          * Returns true if display is not "none"
7568          * @return {Boolean}
7569          */
7570         isDisplayed : function() {
7571             return this.getStyle("display") != "none";
7572         },
7573
7574         /**
7575          * Toggles the element's visibility or display, depending on visibility mode.
7576          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7577          * @return {Roo.Element} this
7578          */
7579         toggle : function(animate){
7580             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7581             return this;
7582         },
7583
7584         /**
7585          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7586          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7587          * @return {Roo.Element} this
7588          */
7589         setDisplayed : function(value) {
7590             if(typeof value == "boolean"){
7591                value = value ? this.originalDisplay : "none";
7592             }
7593             this.setStyle("display", value);
7594             return this;
7595         },
7596
7597         /**
7598          * Tries to focus the element. Any exceptions are caught and ignored.
7599          * @return {Roo.Element} this
7600          */
7601         focus : function() {
7602             try{
7603                 this.dom.focus();
7604             }catch(e){}
7605             return this;
7606         },
7607
7608         /**
7609          * Tries to blur the element. Any exceptions are caught and ignored.
7610          * @return {Roo.Element} this
7611          */
7612         blur : function() {
7613             try{
7614                 this.dom.blur();
7615             }catch(e){}
7616             return this;
7617         },
7618
7619         /**
7620          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7621          * @param {String/Array} className The CSS class to add, or an array of classes
7622          * @return {Roo.Element} this
7623          */
7624         addClass : function(className){
7625             if(className instanceof Array){
7626                 for(var i = 0, len = className.length; i < len; i++) {
7627                     this.addClass(className[i]);
7628                 }
7629             }else{
7630                 if(className && !this.hasClass(className)){
7631                     this.dom.className = this.dom.className + " " + className;
7632                 }
7633             }
7634             return this;
7635         },
7636
7637         /**
7638          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7639          * @param {String/Array} className The CSS class to add, or an array of classes
7640          * @return {Roo.Element} this
7641          */
7642         radioClass : function(className){
7643             var siblings = this.dom.parentNode.childNodes;
7644             for(var i = 0; i < siblings.length; i++) {
7645                 var s = siblings[i];
7646                 if(s.nodeType == 1){
7647                     Roo.get(s).removeClass(className);
7648                 }
7649             }
7650             this.addClass(className);
7651             return this;
7652         },
7653
7654         /**
7655          * Removes one or more CSS classes from the element.
7656          * @param {String/Array} className The CSS class to remove, or an array of classes
7657          * @return {Roo.Element} this
7658          */
7659         removeClass : function(className){
7660             if(!className || !this.dom.className){
7661                 return this;
7662             }
7663             if(className instanceof Array){
7664                 for(var i = 0, len = className.length; i < len; i++) {
7665                     this.removeClass(className[i]);
7666                 }
7667             }else{
7668                 if(this.hasClass(className)){
7669                     var re = this.classReCache[className];
7670                     if (!re) {
7671                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7672                        this.classReCache[className] = re;
7673                     }
7674                     this.dom.className =
7675                         this.dom.className.replace(re, " ");
7676                 }
7677             }
7678             return this;
7679         },
7680
7681         // private
7682         classReCache: {},
7683
7684         /**
7685          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7686          * @param {String} className The CSS class to toggle
7687          * @return {Roo.Element} this
7688          */
7689         toggleClass : function(className){
7690             if(this.hasClass(className)){
7691                 this.removeClass(className);
7692             }else{
7693                 this.addClass(className);
7694             }
7695             return this;
7696         },
7697
7698         /**
7699          * Checks if the specified CSS class exists on this element's DOM node.
7700          * @param {String} className The CSS class to check for
7701          * @return {Boolean} True if the class exists, else false
7702          */
7703         hasClass : function(className){
7704             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7705         },
7706
7707         /**
7708          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7709          * @param {String} oldClassName The CSS class to replace
7710          * @param {String} newClassName The replacement CSS class
7711          * @return {Roo.Element} this
7712          */
7713         replaceClass : function(oldClassName, newClassName){
7714             this.removeClass(oldClassName);
7715             this.addClass(newClassName);
7716             return this;
7717         },
7718
7719         /**
7720          * Returns an object with properties matching the styles requested.
7721          * For example, el.getStyles('color', 'font-size', 'width') might return
7722          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7723          * @param {String} style1 A style name
7724          * @param {String} style2 A style name
7725          * @param {String} etc.
7726          * @return {Object} The style object
7727          */
7728         getStyles : function(){
7729             var a = arguments, len = a.length, r = {};
7730             for(var i = 0; i < len; i++){
7731                 r[a[i]] = this.getStyle(a[i]);
7732             }
7733             return r;
7734         },
7735
7736         /**
7737          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7738          * @param {String} property The style property whose value is returned.
7739          * @return {String} The current value of the style property for this element.
7740          */
7741         getStyle : function(){
7742             return view && view.getComputedStyle ?
7743                 function(prop){
7744                     var el = this.dom, v, cs, camel;
7745                     if(prop == 'float'){
7746                         prop = "cssFloat";
7747                     }
7748                     if(el.style && (v = el.style[prop])){
7749                         return v;
7750                     }
7751                     if(cs = view.getComputedStyle(el, "")){
7752                         if(!(camel = propCache[prop])){
7753                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7754                         }
7755                         return cs[camel];
7756                     }
7757                     return null;
7758                 } :
7759                 function(prop){
7760                     var el = this.dom, v, cs, camel;
7761                     if(prop == 'opacity'){
7762                         if(typeof el.style.filter == 'string'){
7763                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7764                             if(m){
7765                                 var fv = parseFloat(m[1]);
7766                                 if(!isNaN(fv)){
7767                                     return fv ? fv / 100 : 0;
7768                                 }
7769                             }
7770                         }
7771                         return 1;
7772                     }else if(prop == 'float'){
7773                         prop = "styleFloat";
7774                     }
7775                     if(!(camel = propCache[prop])){
7776                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7777                     }
7778                     if(v = el.style[camel]){
7779                         return v;
7780                     }
7781                     if(cs = el.currentStyle){
7782                         return cs[camel];
7783                     }
7784                     return null;
7785                 };
7786         }(),
7787
7788         /**
7789          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7790          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7791          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7792          * @return {Roo.Element} this
7793          */
7794         setStyle : function(prop, value){
7795             if(typeof prop == "string"){
7796                 
7797                 if (prop == 'float') {
7798                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7799                     return this;
7800                 }
7801                 
7802                 var camel;
7803                 if(!(camel = propCache[prop])){
7804                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7805                 }
7806                 
7807                 if(camel == 'opacity') {
7808                     this.setOpacity(value);
7809                 }else{
7810                     this.dom.style[camel] = value;
7811                 }
7812             }else{
7813                 for(var style in prop){
7814                     if(typeof prop[style] != "function"){
7815                        this.setStyle(style, prop[style]);
7816                     }
7817                 }
7818             }
7819             return this;
7820         },
7821
7822         /**
7823          * More flexible version of {@link #setStyle} for setting style properties.
7824          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7825          * a function which returns such a specification.
7826          * @return {Roo.Element} this
7827          */
7828         applyStyles : function(style){
7829             Roo.DomHelper.applyStyles(this.dom, style);
7830             return this;
7831         },
7832
7833         /**
7834           * Gets the current X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7835           * @return {Number} The X position of the element
7836           */
7837         getX : function(){
7838             return D.getX(this.dom);
7839         },
7840
7841         /**
7842           * Gets the current Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7843           * @return {Number} The Y position of the element
7844           */
7845         getY : function(){
7846             return D.getY(this.dom);
7847         },
7848
7849         /**
7850           * Gets the current position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7851           * @return {Array} The XY position of the element
7852           */
7853         getXY : function(){
7854             return D.getXY(this.dom);
7855         },
7856
7857         /**
7858          * Sets the X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7859          * @param {Number} The X position of the element
7860          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7861          * @return {Roo.Element} this
7862          */
7863         setX : function(x, animate){
7864             if(!animate || !A){
7865                 D.setX(this.dom, x);
7866             }else{
7867                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7868             }
7869             return this;
7870         },
7871
7872         /**
7873          * Sets the Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7874          * @param {Number} The Y position of the element
7875          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7876          * @return {Roo.Element} this
7877          */
7878         setY : function(y, animate){
7879             if(!animate || !A){
7880                 D.setY(this.dom, y);
7881             }else{
7882                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7883             }
7884             return this;
7885         },
7886
7887         /**
7888          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7889          * @param {String} left The left CSS property value
7890          * @return {Roo.Element} this
7891          */
7892         setLeft : function(left){
7893             this.setStyle("left", this.addUnits(left));
7894             return this;
7895         },
7896
7897         /**
7898          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7899          * @param {String} top The top CSS property value
7900          * @return {Roo.Element} this
7901          */
7902         setTop : function(top){
7903             this.setStyle("top", this.addUnits(top));
7904             return this;
7905         },
7906
7907         /**
7908          * Sets the element's CSS right style.
7909          * @param {String} right The right CSS property value
7910          * @return {Roo.Element} this
7911          */
7912         setRight : function(right){
7913             this.setStyle("right", this.addUnits(right));
7914             return this;
7915         },
7916
7917         /**
7918          * Sets the element's CSS bottom style.
7919          * @param {String} bottom The bottom CSS property value
7920          * @return {Roo.Element} this
7921          */
7922         setBottom : function(bottom){
7923             this.setStyle("bottom", this.addUnits(bottom));
7924             return this;
7925         },
7926
7927         /**
7928          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7929          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7930          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7931          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7932          * @return {Roo.Element} this
7933          */
7934         setXY : function(pos, animate){
7935             if(!animate || !A){
7936                 D.setXY(this.dom, pos);
7937             }else{
7938                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7939             }
7940             return this;
7941         },
7942
7943         /**
7944          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7945          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7946          * @param {Number} x X value for new position (coordinates are page-based)
7947          * @param {Number} y Y value for new position (coordinates are page-based)
7948          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7949          * @return {Roo.Element} this
7950          */
7951         setLocation : function(x, y, animate){
7952             this.setXY([x, y], this.preanim(arguments, 2));
7953             return this;
7954         },
7955
7956         /**
7957          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7958          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7959          * @param {Number} x X value for new position (coordinates are page-based)
7960          * @param {Number} y Y value for new position (coordinates are page-based)
7961          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7962          * @return {Roo.Element} this
7963          */
7964         moveTo : function(x, y, animate){
7965             this.setXY([x, y], this.preanim(arguments, 2));
7966             return this;
7967         },
7968
7969         /**
7970          * Returns the region of the given element.
7971          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7972          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7973          */
7974         getRegion : function(){
7975             return D.getRegion(this.dom);
7976         },
7977
7978         /**
7979          * Returns the offset height of the element
7980          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7981          * @return {Number} The element's height
7982          */
7983         getHeight : function(contentHeight){
7984             var h = this.dom.offsetHeight || 0;
7985             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7986         },
7987
7988         /**
7989          * Returns the offset width of the element
7990          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7991          * @return {Number} The element's width
7992          */
7993         getWidth : function(contentWidth){
7994             var w = this.dom.offsetWidth || 0;
7995             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7996         },
7997
7998         /**
7999          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
8000          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
8001          * if a height has not been set using CSS.
8002          * @return {Number}
8003          */
8004         getComputedHeight : function(){
8005             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8006             if(!h){
8007                 h = parseInt(this.getStyle('height'), 10) || 0;
8008                 if(!this.isBorderBox()){
8009                     h += this.getFrameWidth('tb');
8010                 }
8011             }
8012             return h;
8013         },
8014
8015         /**
8016          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8017          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8018          * if a width has not been set using CSS.
8019          * @return {Number}
8020          */
8021         getComputedWidth : function(){
8022             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8023             if(!w){
8024                 w = parseInt(this.getStyle('width'), 10) || 0;
8025                 if(!this.isBorderBox()){
8026                     w += this.getFrameWidth('lr');
8027                 }
8028             }
8029             return w;
8030         },
8031
8032         /**
8033          * Returns the size of the element.
8034          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8035          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8036          */
8037         getSize : function(contentSize){
8038             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8039         },
8040
8041         /**
8042          * Returns the width and height of the viewport.
8043          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8044          */
8045         getViewSize : function(){
8046             var d = this.dom, doc = document, aw = 0, ah = 0;
8047             if(d == doc || d == doc.body){
8048                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8049             }else{
8050                 return {
8051                     width : d.clientWidth,
8052                     height: d.clientHeight
8053                 };
8054             }
8055         },
8056
8057         /**
8058          * Returns the value of the "value" attribute
8059          * @param {Boolean} asNumber true to parse the value as a number
8060          * @return {String/Number}
8061          */
8062         getValue : function(asNumber){
8063             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8064         },
8065
8066         // private
8067         adjustWidth : function(width){
8068             if(typeof width == "number"){
8069                 if(this.autoBoxAdjust && !this.isBorderBox()){
8070                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8071                 }
8072                 if(width < 0){
8073                     width = 0;
8074                 }
8075             }
8076             return width;
8077         },
8078
8079         // private
8080         adjustHeight : function(height){
8081             if(typeof height == "number"){
8082                if(this.autoBoxAdjust && !this.isBorderBox()){
8083                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8084                }
8085                if(height < 0){
8086                    height = 0;
8087                }
8088             }
8089             return height;
8090         },
8091
8092         /**
8093          * Set the width of the element
8094          * @param {Number} width The new width
8095          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8096          * @return {Roo.Element} this
8097          */
8098         setWidth : function(width, animate){
8099             width = this.adjustWidth(width);
8100             if(!animate || !A){
8101                 this.dom.style.width = this.addUnits(width);
8102             }else{
8103                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8104             }
8105             return this;
8106         },
8107
8108         /**
8109          * Set the height of the element
8110          * @param {Number} height The new height
8111          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8112          * @return {Roo.Element} this
8113          */
8114          setHeight : function(height, animate){
8115             height = this.adjustHeight(height);
8116             if(!animate || !A){
8117                 this.dom.style.height = this.addUnits(height);
8118             }else{
8119                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8120             }
8121             return this;
8122         },
8123
8124         /**
8125          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8126          * @param {Number} width The new width
8127          * @param {Number} height The new height
8128          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8129          * @return {Roo.Element} this
8130          */
8131          setSize : function(width, height, animate){
8132             if(typeof width == "object"){ // in case of object from getSize()
8133                 height = width.height; width = width.width;
8134             }
8135             width = this.adjustWidth(width); height = this.adjustHeight(height);
8136             if(!animate || !A){
8137                 this.dom.style.width = this.addUnits(width);
8138                 this.dom.style.height = this.addUnits(height);
8139             }else{
8140                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8141             }
8142             return this;
8143         },
8144
8145         /**
8146          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8147          * @param {Number} x X value for new position (coordinates are page-based)
8148          * @param {Number} y Y value for new position (coordinates are page-based)
8149          * @param {Number} width The new width
8150          * @param {Number} height The new height
8151          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8152          * @return {Roo.Element} this
8153          */
8154         setBounds : function(x, y, width, height, animate){
8155             if(!animate || !A){
8156                 this.setSize(width, height);
8157                 this.setLocation(x, y);
8158             }else{
8159                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8160                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8161                               this.preanim(arguments, 4), 'motion');
8162             }
8163             return this;
8164         },
8165
8166         /**
8167          * Sets the element's position and size the the specified region. If animation is true then width, height, x and y will be animated concurrently.
8168          * @param {Roo.lib.Region} region The region to fill
8169          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8170          * @return {Roo.Element} this
8171          */
8172         setRegion : function(region, animate){
8173             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8174             return this;
8175         },
8176
8177         /**
8178          * Appends an event handler
8179          *
8180          * @param {String}   eventName     The type of event to append
8181          * @param {Function} fn        The method the event invokes
8182          * @param {Object} scope       (optional) The scope (this object) of the fn
8183          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8184          */
8185         addListener : function(eventName, fn, scope, options){
8186             if (this.dom) {
8187                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8188             }
8189         },
8190
8191         /**
8192          * Removes an event handler from this element
8193          * @param {String} eventName the type of event to remove
8194          * @param {Function} fn the method the event invokes
8195          * @return {Roo.Element} this
8196          */
8197         removeListener : function(eventName, fn){
8198             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8199             return this;
8200         },
8201
8202         /**
8203          * Removes all previous added listeners from this element
8204          * @return {Roo.Element} this
8205          */
8206         removeAllListeners : function(){
8207             E.purgeElement(this.dom);
8208             return this;
8209         },
8210
8211         relayEvent : function(eventName, observable){
8212             this.on(eventName, function(e){
8213                 observable.fireEvent(eventName, e);
8214             });
8215         },
8216
8217         /**
8218          * Set the opacity of the element
8219          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8220          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8221          * @return {Roo.Element} this
8222          */
8223          setOpacity : function(opacity, animate){
8224             if(!animate || !A){
8225                 var s = this.dom.style;
8226                 if(Roo.isIE){
8227                     s.zoom = 1;
8228                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8229                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8230                 }else{
8231                     s.opacity = opacity;
8232                 }
8233             }else{
8234                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8235             }
8236             return this;
8237         },
8238
8239         /**
8240          * Gets the left X coordinate
8241          * @param {Boolean} local True to get the local css position instead of page coordinate
8242          * @return {Number}
8243          */
8244         getLeft : function(local){
8245             if(!local){
8246                 return this.getX();
8247             }else{
8248                 return parseInt(this.getStyle("left"), 10) || 0;
8249             }
8250         },
8251
8252         /**
8253          * Gets the right X coordinate of the element (element X position + element width)
8254          * @param {Boolean} local True to get the local css position instead of page coordinate
8255          * @return {Number}
8256          */
8257         getRight : function(local){
8258             if(!local){
8259                 return this.getX() + this.getWidth();
8260             }else{
8261                 return (this.getLeft(true) + this.getWidth()) || 0;
8262             }
8263         },
8264
8265         /**
8266          * Gets the top Y coordinate
8267          * @param {Boolean} local True to get the local css position instead of page coordinate
8268          * @return {Number}
8269          */
8270         getTop : function(local) {
8271             if(!local){
8272                 return this.getY();
8273             }else{
8274                 return parseInt(this.getStyle("top"), 10) || 0;
8275             }
8276         },
8277
8278         /**
8279          * Gets the bottom Y coordinate of the element (element Y position + element height)
8280          * @param {Boolean} local True to get the local css position instead of page coordinate
8281          * @return {Number}
8282          */
8283         getBottom : function(local){
8284             if(!local){
8285                 return this.getY() + this.getHeight();
8286             }else{
8287                 return (this.getTop(true) + this.getHeight()) || 0;
8288             }
8289         },
8290
8291         /**
8292         * Initializes positioning on this element. If a desired position is not passed, it will make the
8293         * the element positioned relative IF it is not already positioned.
8294         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8295         * @param {Number} zIndex (optional) The zIndex to apply
8296         * @param {Number} x (optional) Set the page X position
8297         * @param {Number} y (optional) Set the page Y position
8298         */
8299         position : function(pos, zIndex, x, y){
8300             if(!pos){
8301                if(this.getStyle('position') == 'static'){
8302                    this.setStyle('position', 'relative');
8303                }
8304             }else{
8305                 this.setStyle("position", pos);
8306             }
8307             if(zIndex){
8308                 this.setStyle("z-index", zIndex);
8309             }
8310             if(x !== undefined && y !== undefined){
8311                 this.setXY([x, y]);
8312             }else if(x !== undefined){
8313                 this.setX(x);
8314             }else if(y !== undefined){
8315                 this.setY(y);
8316             }
8317         },
8318
8319         /**
8320         * Clear positioning back to the default when the document was loaded
8321         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8322         * @return {Roo.Element} this
8323          */
8324         clearPositioning : function(value){
8325             value = value ||'';
8326             this.setStyle({
8327                 "left": value,
8328                 "right": value,
8329                 "top": value,
8330                 "bottom": value,
8331                 "z-index": "",
8332                 "position" : "static"
8333             });
8334             return this;
8335         },
8336
8337         /**
8338         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8339         * snapshot before performing an update and then restoring the element.
8340         * @return {Object}
8341         */
8342         getPositioning : function(){
8343             var l = this.getStyle("left");
8344             var t = this.getStyle("top");
8345             return {
8346                 "position" : this.getStyle("position"),
8347                 "left" : l,
8348                 "right" : l ? "" : this.getStyle("right"),
8349                 "top" : t,
8350                 "bottom" : t ? "" : this.getStyle("bottom"),
8351                 "z-index" : this.getStyle("z-index")
8352             };
8353         },
8354
8355         /**
8356          * Gets the width of the border(s) for the specified side(s)
8357          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8358          * passing lr would get the border (l)eft width + the border (r)ight width.
8359          * @return {Number} The width of the sides passed added together
8360          */
8361         getBorderWidth : function(side){
8362             return this.addStyles(side, El.borders);
8363         },
8364
8365         /**
8366          * Gets the width of the padding(s) for the specified side(s)
8367          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8368          * passing lr would get the padding (l)eft + the padding (r)ight.
8369          * @return {Number} The padding of the sides passed added together
8370          */
8371         getPadding : function(side){
8372             return this.addStyles(side, El.paddings);
8373         },
8374
8375         /**
8376         * Set positioning with an object returned by getPositioning().
8377         * @param {Object} posCfg
8378         * @return {Roo.Element} this
8379          */
8380         setPositioning : function(pc){
8381             this.applyStyles(pc);
8382             if(pc.right == "auto"){
8383                 this.dom.style.right = "";
8384             }
8385             if(pc.bottom == "auto"){
8386                 this.dom.style.bottom = "";
8387             }
8388             return this;
8389         },
8390
8391         // private
8392         fixDisplay : function(){
8393             if(this.getStyle("display") == "none"){
8394                 this.setStyle("visibility", "hidden");
8395                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8396                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8397                     this.setStyle("display", "block");
8398                 }
8399             }
8400         },
8401
8402         /**
8403          * Quick set left and top adding default units
8404          * @param {String} left The left CSS property value
8405          * @param {String} top The top CSS property value
8406          * @return {Roo.Element} this
8407          */
8408          setLeftTop : function(left, top){
8409             this.dom.style.left = this.addUnits(left);
8410             this.dom.style.top = this.addUnits(top);
8411             return this;
8412         },
8413
8414         /**
8415          * Move this element relative to its current position.
8416          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8417          * @param {Number} distance How far to move the element in pixels
8418          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8419          * @return {Roo.Element} this
8420          */
8421          move : function(direction, distance, animate){
8422             var xy = this.getXY();
8423             direction = direction.toLowerCase();
8424             switch(direction){
8425                 case "l":
8426                 case "left":
8427                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8428                     break;
8429                case "r":
8430                case "right":
8431                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8432                     break;
8433                case "t":
8434                case "top":
8435                case "up":
8436                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8437                     break;
8438                case "b":
8439                case "bottom":
8440                case "down":
8441                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8442                     break;
8443             }
8444             return this;
8445         },
8446
8447         /**
8448          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8449          * @return {Roo.Element} this
8450          */
8451         clip : function(){
8452             if(!this.isClipped){
8453                this.isClipped = true;
8454                this.originalClip = {
8455                    "o": this.getStyle("overflow"),
8456                    "x": this.getStyle("overflow-x"),
8457                    "y": this.getStyle("overflow-y")
8458                };
8459                this.setStyle("overflow", "hidden");
8460                this.setStyle("overflow-x", "hidden");
8461                this.setStyle("overflow-y", "hidden");
8462             }
8463             return this;
8464         },
8465
8466         /**
8467          *  Return clipping (overflow) to original clipping before clip() was called
8468          * @return {Roo.Element} this
8469          */
8470         unclip : function(){
8471             if(this.isClipped){
8472                 this.isClipped = false;
8473                 var o = this.originalClip;
8474                 if(o.o){this.setStyle("overflow", o.o);}
8475                 if(o.x){this.setStyle("overflow-x", o.x);}
8476                 if(o.y){this.setStyle("overflow-y", o.y);}
8477             }
8478             return this;
8479         },
8480
8481
8482         /**
8483          * Gets the x,y coordinates specified by the anchor position on the element.
8484          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8485          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8486          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8487          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8488          * @return {Array} [x, y] An array containing the element's x and y coordinates
8489          */
8490         getAnchorXY : function(anchor, local, s){
8491             //Passing a different size is useful for pre-calculating anchors,
8492             //especially for anchored animations that change the el size.
8493
8494             var w, h, vp = false;
8495             if(!s){
8496                 var d = this.dom;
8497                 if(d == document.body || d == document){
8498                     vp = true;
8499                     w = D.getViewWidth(); h = D.getViewHeight();
8500                 }else{
8501                     w = this.getWidth(); h = this.getHeight();
8502                 }
8503             }else{
8504                 w = s.width;  h = s.height;
8505             }
8506             var x = 0, y = 0, r = Math.round;
8507             switch((anchor || "tl").toLowerCase()){
8508                 case "c":
8509                     x = r(w*.5);
8510                     y = r(h*.5);
8511                 break;
8512                 case "t":
8513                     x = r(w*.5);
8514                     y = 0;
8515                 break;
8516                 case "l":
8517                     x = 0;
8518                     y = r(h*.5);
8519                 break;
8520                 case "r":
8521                     x = w;
8522                     y = r(h*.5);
8523                 break;
8524                 case "b":
8525                     x = r(w*.5);
8526                     y = h;
8527                 break;
8528                 case "tl":
8529                     x = 0;
8530                     y = 0;
8531                 break;
8532                 case "bl":
8533                     x = 0;
8534                     y = h;
8535                 break;
8536                 case "br":
8537                     x = w;
8538                     y = h;
8539                 break;
8540                 case "tr":
8541                     x = w;
8542                     y = 0;
8543                 break;
8544             }
8545             if(local === true){
8546                 return [x, y];
8547             }
8548             if(vp){
8549                 var sc = this.getScroll();
8550                 return [x + sc.left, y + sc.top];
8551             }
8552             //Add the element's offset xy
8553             var o = this.getXY();
8554             return [x+o[0], y+o[1]];
8555         },
8556
8557         /**
8558          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8559          * supported position values.
8560          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8561          * @param {String} position The position to align to.
8562          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8563          * @return {Array} [x, y]
8564          */
8565         getAlignToXY : function(el, p, o){
8566             el = Roo.get(el);
8567             var d = this.dom;
8568             if(!el.dom){
8569                 throw "Element.alignTo with an element that doesn't exist";
8570             }
8571             var c = false; //constrain to viewport
8572             var p1 = "", p2 = "";
8573             o = o || [0,0];
8574
8575             if(!p){
8576                 p = "tl-bl";
8577             }else if(p == "?"){
8578                 p = "tl-bl?";
8579             }else if(p.indexOf("-") == -1){
8580                 p = "tl-" + p;
8581             }
8582             p = p.toLowerCase();
8583             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8584             if(!m){
8585                throw "Element.alignTo with an invalid alignment " + p;
8586             }
8587             p1 = m[1]; p2 = m[2]; c = !!m[3];
8588
8589             //Subtract the aligned el's internal xy from the target's offset xy
8590             //plus custom offset to get the aligned el's new offset xy
8591             var a1 = this.getAnchorXY(p1, true);
8592             var a2 = el.getAnchorXY(p2, false);
8593             var x = a2[0] - a1[0] + o[0];
8594             var y = a2[1] - a1[1] + o[1];
8595             if(c){
8596                 //constrain the aligned el to viewport if necessary
8597                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8598                 // 5px of margin for ie
8599                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8600
8601                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8602                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8603                 //otherwise swap the aligned el to the opposite border of the target.
8604                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8605                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8606                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8607                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8608
8609                var doc = document;
8610                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8611                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8612
8613                if((x+w) > dw + scrollX){
8614                     x = swapX ? r.left-w : dw+scrollX-w;
8615                 }
8616                if(x < scrollX){
8617                    x = swapX ? r.right : scrollX;
8618                }
8619                if((y+h) > dh + scrollY){
8620                     y = swapY ? r.top-h : dh+scrollY-h;
8621                 }
8622                if (y < scrollY){
8623                    y = swapY ? r.bottom : scrollY;
8624                }
8625             }
8626             return [x,y];
8627         },
8628
8629         // private
8630         getConstrainToXY : function(){
8631             var os = {top:0, left:0, bottom:0, right: 0};
8632
8633             return function(el, local, offsets, proposedXY){
8634                 el = Roo.get(el);
8635                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8636
8637                 var vw, vh, vx = 0, vy = 0;
8638                 if(el.dom == document.body || el.dom == document){
8639                     vw = Roo.lib.Dom.getViewWidth();
8640                     vh = Roo.lib.Dom.getViewHeight();
8641                 }else{
8642                     vw = el.dom.clientWidth;
8643                     vh = el.dom.clientHeight;
8644                     if(!local){
8645                         var vxy = el.getXY();
8646                         vx = vxy[0];
8647                         vy = vxy[1];
8648                     }
8649                 }
8650
8651                 var s = el.getScroll();
8652
8653                 vx += offsets.left + s.left;
8654                 vy += offsets.top + s.top;
8655
8656                 vw -= offsets.right;
8657                 vh -= offsets.bottom;
8658
8659                 var vr = vx+vw;
8660                 var vb = vy+vh;
8661
8662                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8663                 var x = xy[0], y = xy[1];
8664                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8665
8666                 // only move it if it needs it
8667                 var moved = false;
8668
8669                 // first validate right/bottom
8670                 if((x + w) > vr){
8671                     x = vr - w;
8672                     moved = true;
8673                 }
8674                 if((y + h) > vb){
8675                     y = vb - h;
8676                     moved = true;
8677                 }
8678                 // then make sure top/left isn't negative
8679                 if(x < vx){
8680                     x = vx;
8681                     moved = true;
8682                 }
8683                 if(y < vy){
8684                     y = vy;
8685                     moved = true;
8686                 }
8687                 return moved ? [x, y] : false;
8688             };
8689         }(),
8690
8691         // private
8692         adjustForConstraints : function(xy, parent, offsets){
8693             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8694         },
8695
8696         /**
8697          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8698          * document it aligns it to the viewport.
8699          * The position parameter is optional, and can be specified in any one of the following formats:
8700          * <ul>
8701          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8702          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8703          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8704          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8705          *   <li><b>Two anchors</b>: If two values from the table below are passed separated by a dash, the first value is used as the
8706          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8707          * </ul>
8708          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8709          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8710          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8711          * that specified in order to enforce the viewport constraints.
8712          * Following are all of the supported anchor positions:
8713     <pre>
8714     Value  Description
8715     -----  -----------------------------
8716     tl     The top left corner (default)
8717     t      The center of the top edge
8718     tr     The top right corner
8719     l      The center of the left edge
8720     c      In the center of the element
8721     r      The center of the right edge
8722     bl     The bottom left corner
8723     b      The center of the bottom edge
8724     br     The bottom right corner
8725     </pre>
8726     Example Usage:
8727     <pre><code>
8728     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8729     el.alignTo("other-el");
8730
8731     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8732     el.alignTo("other-el", "tr?");
8733
8734     // align the bottom right corner of el with the center left edge of other-el
8735     el.alignTo("other-el", "br-l?");
8736
8737     // align the center of el with the bottom left corner of other-el and
8738     // adjust the x position by -6 pixels (and the y position by 0)
8739     el.alignTo("other-el", "c-bl", [-6, 0]);
8740     </code></pre>
8741          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8742          * @param {String} position The position to align to.
8743          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8744          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8745          * @return {Roo.Element} this
8746          */
8747         alignTo : function(element, position, offsets, animate){
8748             var xy = this.getAlignToXY(element, position, offsets);
8749             this.setXY(xy, this.preanim(arguments, 3));
8750             return this;
8751         },
8752
8753         /**
8754          * Anchors an element to another element and realigns it when the window is resized.
8755          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8756          * @param {String} position The position to align to.
8757          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8758          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8759          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8760          * is a number, it is used as the buffer delay (defaults to 50ms).
8761          * @param {Function} callback The function to call after the animation finishes
8762          * @return {Roo.Element} this
8763          */
8764         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8765             var action = function(){
8766                 this.alignTo(el, alignment, offsets, animate);
8767                 Roo.callback(callback, this);
8768             };
8769             Roo.EventManager.onWindowResize(action, this);
8770             var tm = typeof monitorScroll;
8771             if(tm != 'undefined'){
8772                 Roo.EventManager.on(window, 'scroll', action, this,
8773                     {buffer: tm == 'number' ? monitorScroll : 50});
8774             }
8775             action.call(this); // align immediately
8776             return this;
8777         },
8778         /**
8779          * Clears any opacity settings from this element. Required in some cases for IE.
8780          * @return {Roo.Element} this
8781          */
8782         clearOpacity : function(){
8783             if (window.ActiveXObject) {
8784                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8785                     this.dom.style.filter = "";
8786                 }
8787             } else {
8788                 this.dom.style.opacity = "";
8789                 this.dom.style["-moz-opacity"] = "";
8790                 this.dom.style["-khtml-opacity"] = "";
8791             }
8792             return this;
8793         },
8794
8795         /**
8796          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8797          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8798          * @return {Roo.Element} this
8799          */
8800         hide : function(animate){
8801             this.setVisible(false, this.preanim(arguments, 0));
8802             return this;
8803         },
8804
8805         /**
8806         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8807         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8808          * @return {Roo.Element} this
8809          */
8810         show : function(animate){
8811             this.setVisible(true, this.preanim(arguments, 0));
8812             return this;
8813         },
8814
8815         /**
8816          * @private Test if size has a unit, otherwise appends the default
8817          */
8818         addUnits : function(size){
8819             return Roo.Element.addUnits(size, this.defaultUnit);
8820         },
8821
8822         /**
8823          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8824          * @return {Roo.Element} this
8825          */
8826         beginMeasure : function(){
8827             var el = this.dom;
8828             if(el.offsetWidth || el.offsetHeight){
8829                 return this; // offsets work already
8830             }
8831             var changed = [];
8832             var p = this.dom, b = document.body; // start with this element
8833             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8834                 var pe = Roo.get(p);
8835                 if(pe.getStyle('display') == 'none'){
8836                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8837                     p.style.visibility = "hidden";
8838                     p.style.display = "block";
8839                 }
8840                 p = p.parentNode;
8841             }
8842             this._measureChanged = changed;
8843             return this;
8844
8845         },
8846
8847         /**
8848          * Restores displays to before beginMeasure was called
8849          * @return {Roo.Element} this
8850          */
8851         endMeasure : function(){
8852             var changed = this._measureChanged;
8853             if(changed){
8854                 for(var i = 0, len = changed.length; i < len; i++) {
8855                     var r = changed[i];
8856                     r.el.style.visibility = r.visibility;
8857                     r.el.style.display = "none";
8858                 }
8859                 this._measureChanged = null;
8860             }
8861             return this;
8862         },
8863
8864         /**
8865         * Update the innerHTML of this element, optionally searching for and processing scripts
8866         * @param {String} html The new HTML
8867         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8868         * @param {Function} callback For async script loading you can be noticed when the update completes
8869         * @return {Roo.Element} this
8870          */
8871         update : function(html, loadScripts, callback){
8872             if(typeof html == "undefined"){
8873                 html = "";
8874             }
8875             if(loadScripts !== true){
8876                 this.dom.innerHTML = html;
8877                 if(typeof callback == "function"){
8878                     callback();
8879                 }
8880                 return this;
8881             }
8882             var id = Roo.id();
8883             var dom = this.dom;
8884
8885             html += '<span id="' + id + '"></span>';
8886
8887             E.onAvailable(id, function(){
8888                 var hd = document.getElementsByTagName("head")[0];
8889                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8890                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8891                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8892
8893                 var match;
8894                 while(match = re.exec(html)){
8895                     var attrs = match[1];
8896                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8897                     if(srcMatch && srcMatch[2]){
8898                        var s = document.createElement("script");
8899                        s.src = srcMatch[2];
8900                        var typeMatch = attrs.match(typeRe);
8901                        if(typeMatch && typeMatch[2]){
8902                            s.type = typeMatch[2];
8903                        }
8904                        hd.appendChild(s);
8905                     }else if(match[2] && match[2].length > 0){
8906                         if(window.execScript) {
8907                            window.execScript(match[2]);
8908                         } else {
8909                             /**
8910                              * eval:var:id
8911                              * eval:var:dom
8912                              * eval:var:html
8913                              * 
8914                              */
8915                            window.eval(match[2]);
8916                         }
8917                     }
8918                 }
8919                 var el = document.getElementById(id);
8920                 if(el){el.parentNode.removeChild(el);}
8921                 if(typeof callback == "function"){
8922                     callback();
8923                 }
8924             });
8925             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8926             return this;
8927         },
8928
8929         /**
8930          * Direct access to the UpdateManager update() method (takes the same parameters).
8931          * @param {String/Function} url The url for this request or a function to call to get the url
8932          * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
8933          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8934          * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
8935          * @return {Roo.Element} this
8936          */
8937         load : function(){
8938             var um = this.getUpdateManager();
8939             um.update.apply(um, arguments);
8940             return this;
8941         },
8942
8943         /**
8944         * Gets this element's UpdateManager
8945         * @return {Roo.UpdateManager} The UpdateManager
8946         */
8947         getUpdateManager : function(){
8948             if(!this.updateManager){
8949                 this.updateManager = new Roo.UpdateManager(this);
8950             }
8951             return this.updateManager;
8952         },
8953
8954         /**
8955          * Disables text selection for this element (normalized across browsers)
8956          * @return {Roo.Element} this
8957          */
8958         unselectable : function(){
8959             this.dom.unselectable = "on";
8960             this.swallowEvent("selectstart", true);
8961             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8962             this.addClass("x-unselectable");
8963             return this;
8964         },
8965
8966         /**
8967         * Calculates the x, y to center this element on the screen
8968         * @return {Array} The x, y values [x, y]
8969         */
8970         getCenterXY : function(){
8971             return this.getAlignToXY(document, 'c-c');
8972         },
8973
8974         /**
8975         * Centers the Element in either the viewport, or another Element.
8976         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8977         */
8978         center : function(centerIn){
8979             this.alignTo(centerIn || document, 'c-c');
8980             return this;
8981         },
8982
8983         /**
8984          * Tests various css rules/browsers to determine if this element uses a border box
8985          * @return {Boolean}
8986          */
8987         isBorderBox : function(){
8988             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8989         },
8990
8991         /**
8992          * Return a box {x, y, width, height} that can be used to set another elements
8993          * size/location to match this element.
8994          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8995          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8996          * @return {Object} box An object in the format {x, y, width, height}
8997          */
8998         getBox : function(contentBox, local){
8999             var xy;
9000             if(!local){
9001                 xy = this.getXY();
9002             }else{
9003                 var left = parseInt(this.getStyle("left"), 10) || 0;
9004                 var top = parseInt(this.getStyle("top"), 10) || 0;
9005                 xy = [left, top];
9006             }
9007             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9008             if(!contentBox){
9009                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9010             }else{
9011                 var l = this.getBorderWidth("l")+this.getPadding("l");
9012                 var r = this.getBorderWidth("r")+this.getPadding("r");
9013                 var t = this.getBorderWidth("t")+this.getPadding("t");
9014                 var b = this.getBorderWidth("b")+this.getPadding("b");
9015                 bx = {x: xy[0]+l, y: xy[1]+t, 0: xy[0]+l, 1: xy[1]+t, width: w-(l+r), height: h-(t+b)};
9016             }
9017             bx.right = bx.x + bx.width;
9018             bx.bottom = bx.y + bx.height;
9019             return bx;
9020         },
9021
9022         /**
9023          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9024          for more information about the sides.
9025          * @param {String} sides
9026          * @return {Number}
9027          */
9028         getFrameWidth : function(sides, onlyContentBox){
9029             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9030         },
9031
9032         /**
9033          * Sets the element's box. Use getBox() on another element to get a box obj. If animate is true then width, height, x and y will be animated concurrently.
9034          * @param {Object} box The box to fill {x, y, width, height}
9035          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9036          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9037          * @return {Roo.Element} this
9038          */
9039         setBox : function(box, adjust, animate){
9040             var w = box.width, h = box.height;
9041             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9042                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9043                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9044             }
9045             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9046             return this;
9047         },
9048
9049         /**
9050          * Forces the browser to repaint this element
9051          * @return {Roo.Element} this
9052          */
9053          repaint : function(){
9054             var dom = this.dom;
9055             this.addClass("x-repaint");
9056             setTimeout(function(){
9057                 Roo.get(dom).removeClass("x-repaint");
9058             }, 1);
9059             return this;
9060         },
9061
9062         /**
9063          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9064          * then it returns the calculated width of the sides (see getPadding)
9065          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9066          * @return {Object/Number}
9067          */
9068         getMargins : function(side){
9069             if(!side){
9070                 return {
9071                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9072                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9073                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9074                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9075                 };
9076             }else{
9077                 return this.addStyles(side, El.margins);
9078              }
9079         },
9080
9081         // private
9082         addStyles : function(sides, styles){
9083             var val = 0, v, w;
9084             for(var i = 0, len = sides.length; i < len; i++){
9085                 v = this.getStyle(styles[sides.charAt(i)]);
9086                 if(v){
9087                      w = parseInt(v, 10);
9088                      if(w){ val += w; }
9089                 }
9090             }
9091             return val;
9092         },
9093
9094         /**
9095          * Creates a proxy element of this element
9096          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9097          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9098          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9099          * @return {Roo.Element} The new proxy element
9100          */
9101         createProxy : function(config, renderTo, matchBox){
9102             if(renderTo){
9103                 renderTo = Roo.getDom(renderTo);
9104             }else{
9105                 renderTo = document.body;
9106             }
9107             config = typeof config == "object" ?
9108                 config : {tag : "div", cls: config};
9109             var proxy = Roo.DomHelper.append(renderTo, config, true);
9110             if(matchBox){
9111                proxy.setBox(this.getBox());
9112             }
9113             return proxy;
9114         },
9115
9116         /**
9117          * Puts a mask over this element to disable user interaction. Requires core.css.
9118          * This method can only be applied to elements which accept child nodes.
9119          * @param {String} msg (optional) A message to display in the mask
9120          * @param {String} msgCls (optional) A css class to apply to the msg element
9121          * @return {Element} The mask  element
9122          */
9123         mask : function(msg, msgCls)
9124         {
9125             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9126                 this.setStyle("position", "relative");
9127             }
9128             if(!this._mask){
9129                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9130             }
9131             
9132             this.addClass("x-masked");
9133             this._mask.setDisplayed(true);
9134             
9135             // we wander
9136             var z = 0;
9137             var dom = this.dom;
9138             while (dom && dom.style) {
9139                 if (!isNaN(parseInt(dom.style.zIndex))) {
9140                     z = Math.max(z, parseInt(dom.style.zIndex));
9141                 }
9142                 dom = dom.parentNode;
9143             }
9144             // if we are masking the body - then it hides everything..
9145             if (this.dom == document.body) {
9146                 z = 1000000;
9147                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9148                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9149             }
9150            
9151             if(typeof msg == 'string'){
9152                 if(!this._maskMsg){
9153                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9154                         cls: "roo-el-mask-msg", 
9155                         cn: [
9156                             {
9157                                 tag: 'i',
9158                                 cls: 'fa fa-spinner fa-spin'
9159                             },
9160                             {
9161                                 tag: 'div'
9162                             }   
9163                         ]
9164                     }, true);
9165                 }
9166                 var mm = this._maskMsg;
9167                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9168                 if (mm.dom.lastChild) { // weird IE issue?
9169                     mm.dom.lastChild.innerHTML = msg;
9170                 }
9171                 mm.setDisplayed(true);
9172                 mm.center(this);
9173                 mm.setStyle('z-index', z + 102);
9174             }
9175             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9176                 this._mask.setHeight(this.getHeight());
9177             }
9178             this._mask.setStyle('z-index', z + 100);
9179             
9180             return this._mask;
9181         },
9182
9183         /**
9184          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9185          * it is cached for reuse.
9186          */
9187         unmask : function(removeEl){
9188             if(this._mask){
9189                 if(removeEl === true){
9190                     this._mask.remove();
9191                     delete this._mask;
9192                     if(this._maskMsg){
9193                         this._maskMsg.remove();
9194                         delete this._maskMsg;
9195                     }
9196                 }else{
9197                     this._mask.setDisplayed(false);
9198                     if(this._maskMsg){
9199                         this._maskMsg.setDisplayed(false);
9200                     }
9201                 }
9202             }
9203             this.removeClass("x-masked");
9204         },
9205
9206         /**
9207          * Returns true if this element is masked
9208          * @return {Boolean}
9209          */
9210         isMasked : function(){
9211             return this._mask && this._mask.isVisible();
9212         },
9213
9214         /**
9215          * Creates an iframe shim for this element to keep selects and other windowed objects from
9216          * showing through.
9217          * @return {Roo.Element} The new shim element
9218          */
9219         createShim : function(){
9220             var el = document.createElement('iframe');
9221             el.frameBorder = 'no';
9222             el.className = 'roo-shim';
9223             if(Roo.isIE && Roo.isSecure){
9224                 el.src = Roo.SSL_SECURE_URL;
9225             }
9226             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9227             shim.autoBoxAdjust = false;
9228             return shim;
9229         },
9230
9231         /**
9232          * Removes this element from the DOM and deletes it from the cache
9233          */
9234         remove : function(){
9235             if(this.dom.parentNode){
9236                 this.dom.parentNode.removeChild(this.dom);
9237             }
9238             delete El.cache[this.dom.id];
9239         },
9240
9241         /**
9242          * Sets up event handlers to add and remove a css class when the mouse is over this element
9243          * @param {String} className
9244          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9245          * mouseout events for children elements
9246          * @return {Roo.Element} this
9247          */
9248         addClassOnOver : function(className, preventFlicker){
9249             this.on("mouseover", function(){
9250                 Roo.fly(this, '_internal').addClass(className);
9251             }, this.dom);
9252             var removeFn = function(e){
9253                 if(preventFlicker !== true || !e.within(this, true)){
9254                     Roo.fly(this, '_internal').removeClass(className);
9255                 }
9256             };
9257             this.on("mouseout", removeFn, this.dom);
9258             return this;
9259         },
9260
9261         /**
9262          * Sets up event handlers to add and remove a css class when this element has the focus
9263          * @param {String} className
9264          * @return {Roo.Element} this
9265          */
9266         addClassOnFocus : function(className){
9267             this.on("focus", function(){
9268                 Roo.fly(this, '_internal').addClass(className);
9269             }, this.dom);
9270             this.on("blur", function(){
9271                 Roo.fly(this, '_internal').removeClass(className);
9272             }, this.dom);
9273             return this;
9274         },
9275         /**
9276          * Sets up event handlers to add and remove a css class when the mouse is down and then up on this element (a click effect)
9277          * @param {String} className
9278          * @return {Roo.Element} this
9279          */
9280         addClassOnClick : function(className){
9281             var dom = this.dom;
9282             this.on("mousedown", function(){
9283                 Roo.fly(dom, '_internal').addClass(className);
9284                 var d = Roo.get(document);
9285                 var fn = function(){
9286                     Roo.fly(dom, '_internal').removeClass(className);
9287                     d.removeListener("mouseup", fn);
9288                 };
9289                 d.on("mouseup", fn);
9290             });
9291             return this;
9292         },
9293
9294         /**
9295          * Stops the specified event from bubbling and optionally prevents the default action
9296          * @param {String} eventName
9297          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9298          * @return {Roo.Element} this
9299          */
9300         swallowEvent : function(eventName, preventDefault){
9301             var fn = function(e){
9302                 e.stopPropagation();
9303                 if(preventDefault){
9304                     e.preventDefault();
9305                 }
9306             };
9307             if(eventName instanceof Array){
9308                 for(var i = 0, len = eventName.length; i < len; i++){
9309                      this.on(eventName[i], fn);
9310                 }
9311                 return this;
9312             }
9313             this.on(eventName, fn);
9314             return this;
9315         },
9316
9317         /**
9318          * @private
9319          */
9320       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9321
9322         /**
9323          * Sizes this element to its parent element's dimensions performing
9324          * neccessary box adjustments.
9325          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9326          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9327          * @return {Roo.Element} this
9328          */
9329         fitToParent : function(monitorResize, targetParent) {
9330           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9331           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9332           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9333             return;
9334           }
9335           var p = Roo.get(targetParent || this.dom.parentNode);
9336           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9337           if (monitorResize === true) {
9338             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9339             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9340           }
9341           return this;
9342         },
9343
9344         /**
9345          * Gets the next sibling, skipping text nodes
9346          * @return {HTMLElement} The next sibling or null
9347          */
9348         getNextSibling : function(){
9349             var n = this.dom.nextSibling;
9350             while(n && n.nodeType != 1){
9351                 n = n.nextSibling;
9352             }
9353             return n;
9354         },
9355
9356         /**
9357          * Gets the previous sibling, skipping text nodes
9358          * @return {HTMLElement} The previous sibling or null
9359          */
9360         getPrevSibling : function(){
9361             var n = this.dom.previousSibling;
9362             while(n && n.nodeType != 1){
9363                 n = n.previousSibling;
9364             }
9365             return n;
9366         },
9367
9368
9369         /**
9370          * Appends the passed element(s) to this element
9371          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9372          * @return {Roo.Element} this
9373          */
9374         appendChild: function(el){
9375             el = Roo.get(el);
9376             el.appendTo(this);
9377             return this;
9378         },
9379
9380         /**
9381          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9382          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9383          * automatically generated with the specified attributes.
9384          * @param {HTMLElement} insertBefore (optional) a child element of this element
9385          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9386          * @return {Roo.Element} The new child element
9387          */
9388         createChild: function(config, insertBefore, returnDom){
9389             config = config || {tag:'div'};
9390             if(insertBefore){
9391                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9392             }
9393             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9394         },
9395
9396         /**
9397          * Appends this element to the passed element
9398          * @param {String/HTMLElement/Element} el The new parent element
9399          * @return {Roo.Element} this
9400          */
9401         appendTo: function(el){
9402             el = Roo.getDom(el);
9403             el.appendChild(this.dom);
9404             return this;
9405         },
9406
9407         /**
9408          * Inserts this element before the passed element in the DOM
9409          * @param {String/HTMLElement/Element} el The element to insert before
9410          * @return {Roo.Element} this
9411          */
9412         insertBefore: function(el){
9413             el = Roo.getDom(el);
9414             el.parentNode.insertBefore(this.dom, el);
9415             return this;
9416         },
9417
9418         /**
9419          * Inserts this element after the passed element in the DOM
9420          * @param {String/HTMLElement/Element} el The element to insert after
9421          * @return {Roo.Element} this
9422          */
9423         insertAfter: function(el){
9424             el = Roo.getDom(el);
9425             el.parentNode.insertBefore(this.dom, el.nextSibling);
9426             return this;
9427         },
9428
9429         /**
9430          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9431          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9432          * @return {Roo.Element} The new child
9433          */
9434         insertFirst: function(el, returnDom){
9435             el = el || {};
9436             if(typeof el == 'object' && !el.nodeType){ // dh config
9437                 return this.createChild(el, this.dom.firstChild, returnDom);
9438             }else{
9439                 el = Roo.getDom(el);
9440                 this.dom.insertBefore(el, this.dom.firstChild);
9441                 return !returnDom ? Roo.get(el) : el;
9442             }
9443         },
9444
9445         /**
9446          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9447          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9448          * @param {String} where (optional) 'before' or 'after' defaults to before
9449          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9450          * @return {Roo.Element} the inserted Element
9451          */
9452         insertSibling: function(el, where, returnDom){
9453             where = where ? where.toLowerCase() : 'before';
9454             el = el || {};
9455             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9456
9457             if(typeof el == 'object' && !el.nodeType){ // dh config
9458                 if(where == 'after' && !this.dom.nextSibling){
9459                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9460                 }else{
9461                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9462                 }
9463
9464             }else{
9465                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9466                             where == 'before' ? this.dom : this.dom.nextSibling);
9467                 if(!returnDom){
9468                     rt = Roo.get(rt);
9469                 }
9470             }
9471             return rt;
9472         },
9473
9474         /**
9475          * Creates and wraps this element with another element
9476          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9477          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9478          * @return {HTMLElement/Element} The newly created wrapper element
9479          */
9480         wrap: function(config, returnDom){
9481             if(!config){
9482                 config = {tag: "div"};
9483             }
9484             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9485             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9486             return newEl;
9487         },
9488
9489         /**
9490          * Replaces the passed element with this element
9491          * @param {String/HTMLElement/Element} el The element to replace
9492          * @return {Roo.Element} this
9493          */
9494         replace: function(el){
9495             el = Roo.get(el);
9496             this.insertBefore(el);
9497             el.remove();
9498             return this;
9499         },
9500
9501         /**
9502          * Inserts an html fragment into this element
9503          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9504          * @param {String} html The HTML fragment
9505          * @param {Boolean} returnEl True to return an Roo.Element
9506          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9507          */
9508         insertHtml : function(where, html, returnEl){
9509             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9510             return returnEl ? Roo.get(el) : el;
9511         },
9512
9513         /**
9514          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9515          * @param {Object} o The object with the attributes
9516          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9517          * @return {Roo.Element} this
9518          */
9519         set : function(o, useSet){
9520             var el = this.dom;
9521             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9522             for(var attr in o){
9523                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9524                 if(attr=="cls"){
9525                     el.className = o["cls"];
9526                 }else{
9527                     if(useSet) {
9528                         el.setAttribute(attr, o[attr]);
9529                     } else {
9530                         el[attr] = o[attr];
9531                     }
9532                 }
9533             }
9534             if(o.style){
9535                 Roo.DomHelper.applyStyles(el, o.style);
9536             }
9537             return this;
9538         },
9539
9540         /**
9541          * Convenience method for constructing a KeyMap
9542          * @param {Number/Array/Object/String} key Either a string with the keys to listen for, the numeric key code, array of key codes or an object with the following options:
9543          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9544          * @param {Function} fn The function to call
9545          * @param {Object} scope (optional) The scope of the function
9546          * @return {Roo.KeyMap} The KeyMap created
9547          */
9548         addKeyListener : function(key, fn, scope){
9549             var config;
9550             if(typeof key != "object" || key instanceof Array){
9551                 config = {
9552                     key: key,
9553                     fn: fn,
9554                     scope: scope
9555                 };
9556             }else{
9557                 config = {
9558                     key : key.key,
9559                     shift : key.shift,
9560                     ctrl : key.ctrl,
9561                     alt : key.alt,
9562                     fn: fn,
9563                     scope: scope
9564                 };
9565             }
9566             return new Roo.KeyMap(this, config);
9567         },
9568
9569         /**
9570          * Creates a KeyMap for this element
9571          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9572          * @return {Roo.KeyMap} The KeyMap created
9573          */
9574         addKeyMap : function(config){
9575             return new Roo.KeyMap(this, config);
9576         },
9577
9578         /**
9579          * Returns true if this element is scrollable.
9580          * @return {Boolean}
9581          */
9582          isScrollable : function(){
9583             var dom = this.dom;
9584             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9585         },
9586
9587         /**
9588          * Scrolls this element the specified scroll point. It does NOT do bounds checking so if you scroll to a weird value it will try to do it. For auto bounds checking, use scroll().
9589          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9590          * @param {Number} value The new scroll value
9591          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9592          * @return {Element} this
9593          */
9594
9595         scrollTo : function(side, value, animate){
9596             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9597             if(!animate || !A){
9598                 this.dom[prop] = value;
9599             }else{
9600                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9601                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9602             }
9603             return this;
9604         },
9605
9606         /**
9607          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9608          * within this element's scrollable range.
9609          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9610          * @param {Number} distance How far to scroll the element in pixels
9611          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9612          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9613          * was scrolled as far as it could go.
9614          */
9615          scroll : function(direction, distance, animate){
9616              if(!this.isScrollable()){
9617                  return;
9618              }
9619              var el = this.dom;
9620              var l = el.scrollLeft, t = el.scrollTop;
9621              var w = el.scrollWidth, h = el.scrollHeight;
9622              var cw = el.clientWidth, ch = el.clientHeight;
9623              direction = direction.toLowerCase();
9624              var scrolled = false;
9625              var a = this.preanim(arguments, 2);
9626              switch(direction){
9627                  case "l":
9628                  case "left":
9629                      if(w - l > cw){
9630                          var v = Math.min(l + distance, w-cw);
9631                          this.scrollTo("left", v, a);
9632                          scrolled = true;
9633                      }
9634                      break;
9635                 case "r":
9636                 case "right":
9637                      if(l > 0){
9638                          var v = Math.max(l - distance, 0);
9639                          this.scrollTo("left", v, a);
9640                          scrolled = true;
9641                      }
9642                      break;
9643                 case "t":
9644                 case "top":
9645                 case "up":
9646                      if(t > 0){
9647                          var v = Math.max(t - distance, 0);
9648                          this.scrollTo("top", v, a);
9649                          scrolled = true;
9650                      }
9651                      break;
9652                 case "b":
9653                 case "bottom":
9654                 case "down":
9655                      if(h - t > ch){
9656                          var v = Math.min(t + distance, h-ch);
9657                          this.scrollTo("top", v, a);
9658                          scrolled = true;
9659                      }
9660                      break;
9661              }
9662              return scrolled;
9663         },
9664
9665         /**
9666          * Translates the passed page coordinates into left/top css values for this element
9667          * @param {Number/Array} x The page x or an array containing [x, y]
9668          * @param {Number} y The page y
9669          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9670          */
9671         translatePoints : function(x, y){
9672             if(typeof x == 'object' || x instanceof Array){
9673                 y = x[1]; x = x[0];
9674             }
9675             var p = this.getStyle('position');
9676             var o = this.getXY();
9677
9678             var l = parseInt(this.getStyle('left'), 10);
9679             var t = parseInt(this.getStyle('top'), 10);
9680
9681             if(isNaN(l)){
9682                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9683             }
9684             if(isNaN(t)){
9685                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9686             }
9687
9688             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9689         },
9690
9691         /**
9692          * Returns the current scroll position of the element.
9693          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9694          */
9695         getScroll : function(){
9696             var d = this.dom, doc = document;
9697             if(d == doc || d == doc.body){
9698                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9699                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9700                 return {left: l, top: t};
9701             }else{
9702                 return {left: d.scrollLeft, top: d.scrollTop};
9703             }
9704         },
9705
9706         /**
9707          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9708          * are convert to standard 6 digit hex color.
9709          * @param {String} attr The css attribute
9710          * @param {String} defaultValue The default value to use when a valid color isn't found
9711          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9712          * YUI color anims.
9713          */
9714         getColor : function(attr, defaultValue, prefix){
9715             var v = this.getStyle(attr);
9716             if(!v || v == "transparent" || v == "inherit") {
9717                 return defaultValue;
9718             }
9719             var color = typeof prefix == "undefined" ? "#" : prefix;
9720             if(v.substr(0, 4) == "rgb("){
9721                 var rvs = v.slice(4, v.length -1).split(",");
9722                 for(var i = 0; i < 3; i++){
9723                     var h = parseInt(rvs[i]).toString(16);
9724                     if(h < 16){
9725                         h = "0" + h;
9726                     }
9727                     color += h;
9728                 }
9729             } else {
9730                 if(v.substr(0, 1) == "#"){
9731                     if(v.length == 4) {
9732                         for(var i = 1; i < 4; i++){
9733                             var c = v.charAt(i);
9734                             color +=  c + c;
9735                         }
9736                     }else if(v.length == 7){
9737                         color += v.substr(1);
9738                     }
9739                 }
9740             }
9741             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9742         },
9743
9744         /**
9745          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9746          * gradient background, rounded corners and a 4-way shadow.
9747          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9748          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9749          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9750          * @return {Roo.Element} this
9751          */
9752         boxWrap : function(cls){
9753             cls = cls || 'x-box';
9754             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9755             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9756             return el;
9757         },
9758
9759         /**
9760          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9761          * @param {String} namespace The namespace in which to look for the attribute
9762          * @param {String} name The attribute name
9763          * @return {String} The attribute value
9764          */
9765         getAttributeNS : Roo.isIE ? function(ns, name){
9766             var d = this.dom;
9767             var type = typeof d[ns+":"+name];
9768             if(type != 'undefined' && type != 'unknown'){
9769                 return d[ns+":"+name];
9770             }
9771             return d[name];
9772         } : function(ns, name){
9773             var d = this.dom;
9774             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9775         },
9776         
9777         
9778         /**
9779          * Sets or Returns the value the dom attribute value
9780          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9781          * @param {String} value (optional) The value to set the attribute to
9782          * @return {String} The attribute value
9783          */
9784         attr : function(name){
9785             if (arguments.length > 1) {
9786                 this.dom.setAttribute(name, arguments[1]);
9787                 return arguments[1];
9788             }
9789             if (typeof(name) == 'object') {
9790                 for(var i in name) {
9791                     this.attr(i, name[i]);
9792                 }
9793                 return name;
9794             }
9795             
9796             
9797             if (!this.dom.hasAttribute(name)) {
9798                 return undefined;
9799             }
9800             return this.dom.getAttribute(name);
9801         }
9802         
9803         
9804         
9805     };
9806
9807     var ep = El.prototype;
9808
9809     /**
9810      * Appends an event handler (Shorthand for addListener)
9811      * @param {String}   eventName     The type of event to append
9812      * @param {Function} fn        The method the event invokes
9813      * @param {Object} scope       (optional) The scope (this object) of the fn
9814      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9815      * @method
9816      */
9817     ep.on = ep.addListener;
9818         // backwards compat
9819     ep.mon = ep.addListener;
9820
9821     /**
9822      * Removes an event handler from this element (shorthand for removeListener)
9823      * @param {String} eventName the type of event to remove
9824      * @param {Function} fn the method the event invokes
9825      * @return {Roo.Element} this
9826      * @method
9827      */
9828     ep.un = ep.removeListener;
9829
9830     /**
9831      * true to automatically adjust width and height settings for box-model issues (default to true)
9832      */
9833     ep.autoBoxAdjust = true;
9834
9835     // private
9836     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9837
9838     // private
9839     El.addUnits = function(v, defaultUnit){
9840         if(v === "" || v == "auto"){
9841             return v;
9842         }
9843         if(v === undefined){
9844             return '';
9845         }
9846         if(typeof v == "number" || !El.unitPattern.test(v)){
9847             return v + (defaultUnit || 'px');
9848         }
9849         return v;
9850     };
9851
9852     // special markup used throughout Roo when box wrapping elements
9853     El.boxMarkup = '<div class="{0}-tl"><div class="{0}-tr"><div class="{0}-tc"></div></div></div><div class="{0}-ml"><div class="{0}-mr"><div class="{0}-mc"></div></div></div><div class="{0}-bl"><div class="{0}-br"><div class="{0}-bc"></div></div></div>';
9854     /**
9855      * Visibility mode constant - Use visibility to hide element
9856      * @static
9857      * @type Number
9858      */
9859     El.VISIBILITY = 1;
9860     /**
9861      * Visibility mode constant - Use display to hide element
9862      * @static
9863      * @type Number
9864      */
9865     El.DISPLAY = 2;
9866
9867     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9868     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9869     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9870
9871
9872
9873     /**
9874      * @private
9875      */
9876     El.cache = {};
9877
9878     var docEl;
9879
9880     /**
9881      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9882      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9883      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9884      * @return {Element} The Element object
9885      * @static
9886      */
9887     El.get = function(el){
9888         var ex, elm, id;
9889         if(!el){ return null; }
9890         if(typeof el == "string"){ // element id
9891             if(!(elm = document.getElementById(el))){
9892                 return null;
9893             }
9894             if(ex = El.cache[el]){
9895                 ex.dom = elm;
9896             }else{
9897                 ex = El.cache[el] = new El(elm);
9898             }
9899             return ex;
9900         }else if(el.tagName){ // dom element
9901             if(!(id = el.id)){
9902                 id = Roo.id(el);
9903             }
9904             if(ex = El.cache[id]){
9905                 ex.dom = el;
9906             }else{
9907                 ex = El.cache[id] = new El(el);
9908             }
9909             return ex;
9910         }else if(el instanceof El){
9911             if(el != docEl){
9912                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9913                                                               // catch case where it hasn't been appended
9914                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9915             }
9916             return el;
9917         }else if(el.isComposite){
9918             return el;
9919         }else if(el instanceof Array){
9920             return El.select(el);
9921         }else if(el == document){
9922             // create a bogus element object representing the document object
9923             if(!docEl){
9924                 var f = function(){};
9925                 f.prototype = El.prototype;
9926                 docEl = new f();
9927                 docEl.dom = document;
9928             }
9929             return docEl;
9930         }
9931         return null;
9932     };
9933
9934     // private
9935     El.uncache = function(el){
9936         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9937             if(a[i]){
9938                 delete El.cache[a[i].id || a[i]];
9939             }
9940         }
9941     };
9942
9943     // private
9944     // Garbage collection - uncache elements/purge listeners on orphaned elements
9945     // so we don't hold a reference and cause the browser to retain them
9946     El.garbageCollect = function(){
9947         if(!Roo.enableGarbageCollector){
9948             clearInterval(El.collectorThread);
9949             return;
9950         }
9951         for(var eid in El.cache){
9952             var el = El.cache[eid], d = el.dom;
9953             // -------------------------------------------------------
9954             // Determining what is garbage:
9955             // -------------------------------------------------------
9956             // !d
9957             // dom node is null, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.parentNode
9960             // no parentNode == direct orphan, definitely garbage
9961             // -------------------------------------------------------
9962             // !d.offsetParent && !document.getElementById(eid)
9963             // display none elements have no offsetParent so we will
9964             // also try to look it up by it's id. However, check
9965             // offsetParent first so we don't do unneeded lookups.
9966             // This enables collection of elements that are not orphans
9967             // directly, but somewhere up the line they have an orphan
9968             // parent.
9969             // -------------------------------------------------------
9970             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9971                 delete El.cache[eid];
9972                 if(d && Roo.enableListenerCollection){
9973                     E.purgeElement(d);
9974                 }
9975             }
9976         }
9977     }
9978     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9979
9980
9981     // dom is optional
9982     El.Flyweight = function(dom){
9983         this.dom = dom;
9984     };
9985     El.Flyweight.prototype = El.prototype;
9986
9987     El._flyweights = {};
9988     /**
9989      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9990      * the dom node can be overwritten by other code.
9991      * @param {String/HTMLElement} el The dom node or id
9992      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9993      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9994      * @static
9995      * @return {Element} The shared Element object
9996      */
9997     El.fly = function(el, named){
9998         named = named || '_global';
9999         el = Roo.getDom(el);
10000         if(!el){
10001             return null;
10002         }
10003         if(!El._flyweights[named]){
10004             El._flyweights[named] = new El.Flyweight();
10005         }
10006         El._flyweights[named].dom = el;
10007         return El._flyweights[named];
10008     };
10009
10010     /**
10011      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10012      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10013      * Shorthand of {@link Roo.Element#get}
10014      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10015      * @return {Element} The Element object
10016      * @member Roo
10017      * @method get
10018      */
10019     Roo.get = El.get;
10020     /**
10021      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10022      * the dom node can be overwritten by other code.
10023      * Shorthand of {@link Roo.Element#fly}
10024      * @param {String/HTMLElement} el The dom node or id
10025      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10026      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10027      * @static
10028      * @return {Element} The shared Element object
10029      * @member Roo
10030      * @method fly
10031      */
10032     Roo.fly = El.fly;
10033
10034     // speedy lookup for elements never to box adjust
10035     var noBoxAdjust = Roo.isStrict ? {
10036         select:1
10037     } : {
10038         input:1, select:1, textarea:1
10039     };
10040     if(Roo.isIE || Roo.isGecko){
10041         noBoxAdjust['button'] = 1;
10042     }
10043
10044
10045     Roo.EventManager.on(window, 'unload', function(){
10046         delete El.cache;
10047         delete El._flyweights;
10048     });
10049 })();
10050
10051
10052
10053
10054 if(Roo.DomQuery){
10055     Roo.Element.selectorFunction = Roo.DomQuery.select;
10056 }
10057
10058 Roo.Element.select = function(selector, unique, root){
10059     var els;
10060     if(typeof selector == "string"){
10061         els = Roo.Element.selectorFunction(selector, root);
10062     }else if(selector.length !== undefined){
10063         els = selector;
10064     }else{
10065         throw "Invalid selector";
10066     }
10067     if(unique === true){
10068         return new Roo.CompositeElement(els);
10069     }else{
10070         return new Roo.CompositeElementLite(els);
10071     }
10072 };
10073 /**
10074  * Selects elements based on the passed CSS selector to enable working on them as 1.
10075  * @param {String/Array} selector The CSS selector or an array of elements
10076  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10077  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10078  * @return {CompositeElementLite/CompositeElement}
10079  * @member Roo
10080  * @method select
10081  */
10082 Roo.select = Roo.Element.select;
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094
10095
10096
10097 /*
10098  * Based on:
10099  * Ext JS Library 1.1.1
10100  * Copyright(c) 2006-2007, Ext JS, LLC.
10101  *
10102  * Originally Released Under LGPL - original licence link has changed is not relivant.
10103  *
10104  * Fork - LGPL
10105  * <script type="text/javascript">
10106  */
10107
10108
10109
10110 //Notifies Element that fx methods are available
10111 Roo.enableFx = true;
10112
10113 /**
10114  * @class Roo.Fx
10115  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10116  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10117  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10118  * Element effects to work.</p><br/>
10119  *
10120  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10121  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10122  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10123  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10124  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10125  * expected results and should be done with care.</p><br/>
10126  *
10127  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10128  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10129 <pre>
10130 Value  Description
10131 -----  -----------------------------
10132 tl     The top left corner
10133 t      The center of the top edge
10134 tr     The top right corner
10135 l      The center of the left edge
10136 r      The center of the right edge
10137 bl     The bottom left corner
10138 b      The center of the bottom edge
10139 br     The bottom right corner
10140 </pre>
10141  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10142  * below are common options that can be passed to any Fx method.</b>
10143  * @cfg {Function} callback A function called when the effect is finished
10144  * @cfg {Object} scope The scope of the effect function
10145  * @cfg {String} easing A valid Easing value for the effect
10146  * @cfg {String} afterCls A css class to apply after the effect
10147  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10148  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10149  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10150  * effects that end with the element being visually hidden, ignored otherwise)
10151  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10152  * a function which returns such a specification that will be applied to the Element after the effect finishes
10153  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10154  * @cfg {Boolean} concurrent Whether to allow subsequently-queued effects to run at the same time as the current effect, or to ensure that they run in sequence
10155  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10156  */
10157 Roo.Fx = {
10158         /**
10159          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10160          * origin for the slide effect.  This function automatically handles wrapping the element with
10161          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10162          * Usage:
10163          *<pre><code>
10164 // default: slide the element in from the top
10165 el.slideIn();
10166
10167 // custom: slide the element in from the right with a 2-second duration
10168 el.slideIn('r', { duration: 2 });
10169
10170 // common config options shown with default values
10171 el.slideIn('t', {
10172     easing: 'easeOut',
10173     duration: .5
10174 });
10175 </code></pre>
10176          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10177          * @param {Object} options (optional) Object literal with any of the Fx config options
10178          * @return {Roo.Element} The Element
10179          */
10180     slideIn : function(anchor, o){
10181         var el = this.getFxEl();
10182         o = o || {};
10183
10184         el.queueFx(o, function(){
10185
10186             anchor = anchor || "t";
10187
10188             // fix display to visibility
10189             this.fixDisplay();
10190
10191             // restore values after effect
10192             var r = this.getFxRestore();
10193             var b = this.getBox();
10194             // fixed size for slide
10195             this.setSize(b);
10196
10197             // wrap if needed
10198             var wrap = this.fxWrap(r.pos, o, "hidden");
10199
10200             var st = this.dom.style;
10201             st.visibility = "visible";
10202             st.position = "absolute";
10203
10204             // clear out temp styles after slide and unwrap
10205             var after = function(){
10206                 el.fxUnwrap(wrap, r.pos, o);
10207                 st.width = r.width;
10208                 st.height = r.height;
10209                 el.afterFx(o);
10210             };
10211             // time to calc the positions
10212             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10213
10214             switch(anchor.toLowerCase()){
10215                 case "t":
10216                     wrap.setSize(b.width, 0);
10217                     st.left = st.bottom = "0";
10218                     a = {height: bh};
10219                 break;
10220                 case "l":
10221                     wrap.setSize(0, b.height);
10222                     st.right = st.top = "0";
10223                     a = {width: bw};
10224                 break;
10225                 case "r":
10226                     wrap.setSize(0, b.height);
10227                     wrap.setX(b.right);
10228                     st.left = st.top = "0";
10229                     a = {width: bw, points: pt};
10230                 break;
10231                 case "b":
10232                     wrap.setSize(b.width, 0);
10233                     wrap.setY(b.bottom);
10234                     st.left = st.top = "0";
10235                     a = {height: bh, points: pt};
10236                 break;
10237                 case "tl":
10238                     wrap.setSize(0, 0);
10239                     st.right = st.bottom = "0";
10240                     a = {width: bw, height: bh};
10241                 break;
10242                 case "bl":
10243                     wrap.setSize(0, 0);
10244                     wrap.setY(b.y+b.height);
10245                     st.right = st.top = "0";
10246                     a = {width: bw, height: bh, points: pt};
10247                 break;
10248                 case "br":
10249                     wrap.setSize(0, 0);
10250                     wrap.setXY([b.right, b.bottom]);
10251                     st.left = st.top = "0";
10252                     a = {width: bw, height: bh, points: pt};
10253                 break;
10254                 case "tr":
10255                     wrap.setSize(0, 0);
10256                     wrap.setX(b.x+b.width);
10257                     st.left = st.bottom = "0";
10258                     a = {width: bw, height: bh, points: pt};
10259                 break;
10260             }
10261             this.dom.style.visibility = "visible";
10262             wrap.show();
10263
10264             arguments.callee.anim = wrap.fxanim(a,
10265                 o,
10266                 'motion',
10267                 .5,
10268                 'easeOut', after);
10269         });
10270         return this;
10271     },
10272     
10273         /**
10274          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10275          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10276          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10277          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10278          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10279          * Usage:
10280          *<pre><code>
10281 // default: slide the element out to the top
10282 el.slideOut();
10283
10284 // custom: slide the element out to the right with a 2-second duration
10285 el.slideOut('r', { duration: 2 });
10286
10287 // common config options shown with default values
10288 el.slideOut('t', {
10289     easing: 'easeOut',
10290     duration: .5,
10291     remove: false,
10292     useDisplay: false
10293 });
10294 </code></pre>
10295          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10296          * @param {Object} options (optional) Object literal with any of the Fx config options
10297          * @return {Roo.Element} The Element
10298          */
10299     slideOut : function(anchor, o){
10300         var el = this.getFxEl();
10301         o = o || {};
10302
10303         el.queueFx(o, function(){
10304
10305             anchor = anchor || "t";
10306
10307             // restore values after effect
10308             var r = this.getFxRestore();
10309             
10310             var b = this.getBox();
10311             // fixed size for slide
10312             this.setSize(b);
10313
10314             // wrap if needed
10315             var wrap = this.fxWrap(r.pos, o, "visible");
10316
10317             var st = this.dom.style;
10318             st.visibility = "visible";
10319             st.position = "absolute";
10320
10321             wrap.setSize(b);
10322
10323             var after = function(){
10324                 if(o.useDisplay){
10325                     el.setDisplayed(false);
10326                 }else{
10327                     el.hide();
10328                 }
10329
10330                 el.fxUnwrap(wrap, r.pos, o);
10331
10332                 st.width = r.width;
10333                 st.height = r.height;
10334
10335                 el.afterFx(o);
10336             };
10337
10338             var a, zero = {to: 0};
10339             switch(anchor.toLowerCase()){
10340                 case "t":
10341                     st.left = st.bottom = "0";
10342                     a = {height: zero};
10343                 break;
10344                 case "l":
10345                     st.right = st.top = "0";
10346                     a = {width: zero};
10347                 break;
10348                 case "r":
10349                     st.left = st.top = "0";
10350                     a = {width: zero, points: {to:[b.right, b.y]}};
10351                 break;
10352                 case "b":
10353                     st.left = st.top = "0";
10354                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10355                 break;
10356                 case "tl":
10357                     st.right = st.bottom = "0";
10358                     a = {width: zero, height: zero};
10359                 break;
10360                 case "bl":
10361                     st.right = st.top = "0";
10362                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10363                 break;
10364                 case "br":
10365                     st.left = st.top = "0";
10366                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10367                 break;
10368                 case "tr":
10369                     st.left = st.bottom = "0";
10370                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10371                 break;
10372             }
10373
10374             arguments.callee.anim = wrap.fxanim(a,
10375                 o,
10376                 'motion',
10377                 .5,
10378                 "easeOut", after);
10379         });
10380         return this;
10381     },
10382
10383         /**
10384          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10385          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10386          * The element must be removed from the DOM using the 'remove' config option if desired.
10387          * Usage:
10388          *<pre><code>
10389 // default
10390 el.puff();
10391
10392 // common config options shown with default values
10393 el.puff({
10394     easing: 'easeOut',
10395     duration: .5,
10396     remove: false,
10397     useDisplay: false
10398 });
10399 </code></pre>
10400          * @param {Object} options (optional) Object literal with any of the Fx config options
10401          * @return {Roo.Element} The Element
10402          */
10403     puff : function(o){
10404         var el = this.getFxEl();
10405         o = o || {};
10406
10407         el.queueFx(o, function(){
10408             this.clearOpacity();
10409             this.show();
10410
10411             // restore values after effect
10412             var r = this.getFxRestore();
10413             var st = this.dom.style;
10414
10415             var after = function(){
10416                 if(o.useDisplay){
10417                     el.setDisplayed(false);
10418                 }else{
10419                     el.hide();
10420                 }
10421
10422                 el.clearOpacity();
10423
10424                 el.setPositioning(r.pos);
10425                 st.width = r.width;
10426                 st.height = r.height;
10427                 st.fontSize = '';
10428                 el.afterFx(o);
10429             };
10430
10431             var width = this.getWidth();
10432             var height = this.getHeight();
10433
10434             arguments.callee.anim = this.fxanim({
10435                     width : {to: this.adjustWidth(width * 2)},
10436                     height : {to: this.adjustHeight(height * 2)},
10437                     points : {by: [-(width * .5), -(height * .5)]},
10438                     opacity : {to: 0},
10439                     fontSize: {to:200, unit: "%"}
10440                 },
10441                 o,
10442                 'motion',
10443                 .5,
10444                 "easeOut", after);
10445         });
10446         return this;
10447     },
10448
10449         /**
10450          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10451          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10452          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10453          * Usage:
10454          *<pre><code>
10455 // default
10456 el.switchOff();
10457
10458 // all config options shown with default values
10459 el.switchOff({
10460     easing: 'easeIn',
10461     duration: .3,
10462     remove: false,
10463     useDisplay: false
10464 });
10465 </code></pre>
10466          * @param {Object} options (optional) Object literal with any of the Fx config options
10467          * @return {Roo.Element} The Element
10468          */
10469     switchOff : function(o){
10470         var el = this.getFxEl();
10471         o = o || {};
10472
10473         el.queueFx(o, function(){
10474             this.clearOpacity();
10475             this.clip();
10476
10477             // restore values after effect
10478             var r = this.getFxRestore();
10479             var st = this.dom.style;
10480
10481             var after = function(){
10482                 if(o.useDisplay){
10483                     el.setDisplayed(false);
10484                 }else{
10485                     el.hide();
10486                 }
10487
10488                 el.clearOpacity();
10489                 el.setPositioning(r.pos);
10490                 st.width = r.width;
10491                 st.height = r.height;
10492
10493                 el.afterFx(o);
10494             };
10495
10496             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10497                 this.clearOpacity();
10498                 (function(){
10499                     this.fxanim({
10500                         height:{to:1},
10501                         points:{by:[0, this.getHeight() * .5]}
10502                     }, o, 'motion', 0.3, 'easeIn', after);
10503                 }).defer(100, this);
10504             });
10505         });
10506         return this;
10507     },
10508
10509     /**
10510      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10511      * changed using the "attr" config option) and then fading back to the original color. If no original
10512      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10513      * Usage:
10514 <pre><code>
10515 // default: highlight background to yellow
10516 el.highlight();
10517
10518 // custom: highlight foreground text to blue for 2 seconds
10519 el.highlight("0000ff", { attr: 'color', duration: 2 });
10520
10521 // common config options shown with default values
10522 el.highlight("ffff9c", {
10523     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10524     endColor: (current color) or "ffffff",
10525     easing: 'easeIn',
10526     duration: 1
10527 });
10528 </code></pre>
10529      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10530      * @param {Object} options (optional) Object literal with any of the Fx config options
10531      * @return {Roo.Element} The Element
10532      */ 
10533     highlight : function(color, o){
10534         var el = this.getFxEl();
10535         o = o || {};
10536
10537         el.queueFx(o, function(){
10538             color = color || "ffff9c";
10539             attr = o.attr || "backgroundColor";
10540
10541             this.clearOpacity();
10542             this.show();
10543
10544             var origColor = this.getColor(attr);
10545             var restoreColor = this.dom.style[attr];
10546             endColor = (o.endColor || origColor) || "ffffff";
10547
10548             var after = function(){
10549                 el.dom.style[attr] = restoreColor;
10550                 el.afterFx(o);
10551             };
10552
10553             var a = {};
10554             a[attr] = {from: color, to: endColor};
10555             arguments.callee.anim = this.fxanim(a,
10556                 o,
10557                 'color',
10558                 1,
10559                 'easeIn', after);
10560         });
10561         return this;
10562     },
10563
10564    /**
10565     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10566     * Usage:
10567 <pre><code>
10568 // default: a single light blue ripple
10569 el.frame();
10570
10571 // custom: 3 red ripples lasting 3 seconds total
10572 el.frame("ff0000", 3, { duration: 3 });
10573
10574 // common config options shown with default values
10575 el.frame("C3DAF9", 1, {
10576     duration: 1 //duration of entire animation (not each individual ripple)
10577     // Note: Easing is not configurable and will be ignored if included
10578 });
10579 </code></pre>
10580     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10581     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10582     * @param {Object} options (optional) Object literal with any of the Fx config options
10583     * @return {Roo.Element} The Element
10584     */
10585     frame : function(color, count, o){
10586         var el = this.getFxEl();
10587         o = o || {};
10588
10589         el.queueFx(o, function(){
10590             color = color || "#C3DAF9";
10591             if(color.length == 6){
10592                 color = "#" + color;
10593             }
10594             count = count || 1;
10595             duration = o.duration || 1;
10596             this.show();
10597
10598             var b = this.getBox();
10599             var animFn = function(){
10600                 var proxy = this.createProxy({
10601
10602                      style:{
10603                         visbility:"hidden",
10604                         position:"absolute",
10605                         "z-index":"35000", // yee haw
10606                         border:"0px solid " + color
10607                      }
10608                   });
10609                 var scale = Roo.isBorderBox ? 2 : 1;
10610                 proxy.animate({
10611                     top:{from:b.y, to:b.y - 20},
10612                     left:{from:b.x, to:b.x - 20},
10613                     borderWidth:{from:0, to:10},
10614                     opacity:{from:1, to:0},
10615                     height:{from:b.height, to:(b.height + (20*scale))},
10616                     width:{from:b.width, to:(b.width + (20*scale))}
10617                 }, duration, function(){
10618                     proxy.remove();
10619                 });
10620                 if(--count > 0){
10621                      animFn.defer((duration/2)*1000, this);
10622                 }else{
10623                     el.afterFx(o);
10624                 }
10625             };
10626             animFn.call(this);
10627         });
10628         return this;
10629     },
10630
10631    /**
10632     * Creates a pause before any subsequent queued effects begin.  If there are
10633     * no effects queued after the pause it will have no effect.
10634     * Usage:
10635 <pre><code>
10636 el.pause(1);
10637 </code></pre>
10638     * @param {Number} seconds The length of time to pause (in seconds)
10639     * @return {Roo.Element} The Element
10640     */
10641     pause : function(seconds){
10642         var el = this.getFxEl();
10643         var o = {};
10644
10645         el.queueFx(o, function(){
10646             setTimeout(function(){
10647                 el.afterFx(o);
10648             }, seconds * 1000);
10649         });
10650         return this;
10651     },
10652
10653    /**
10654     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10655     * using the "endOpacity" config option.
10656     * Usage:
10657 <pre><code>
10658 // default: fade in from opacity 0 to 100%
10659 el.fadeIn();
10660
10661 // custom: fade in from opacity 0 to 75% over 2 seconds
10662 el.fadeIn({ endOpacity: .75, duration: 2});
10663
10664 // common config options shown with default values
10665 el.fadeIn({
10666     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10667     easing: 'easeOut',
10668     duration: .5
10669 });
10670 </code></pre>
10671     * @param {Object} options (optional) Object literal with any of the Fx config options
10672     * @return {Roo.Element} The Element
10673     */
10674     fadeIn : function(o){
10675         var el = this.getFxEl();
10676         o = o || {};
10677         el.queueFx(o, function(){
10678             this.setOpacity(0);
10679             this.fixDisplay();
10680             this.dom.style.visibility = 'visible';
10681             var to = o.endOpacity || 1;
10682             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10683                 o, null, .5, "easeOut", function(){
10684                 if(to == 1){
10685                     this.clearOpacity();
10686                 }
10687                 el.afterFx(o);
10688             });
10689         });
10690         return this;
10691     },
10692
10693    /**
10694     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10695     * using the "endOpacity" config option.
10696     * Usage:
10697 <pre><code>
10698 // default: fade out from the element's current opacity to 0
10699 el.fadeOut();
10700
10701 // custom: fade out from the element's current opacity to 25% over 2 seconds
10702 el.fadeOut({ endOpacity: .25, duration: 2});
10703
10704 // common config options shown with default values
10705 el.fadeOut({
10706     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10707     easing: 'easeOut',
10708     duration: .5
10709     remove: false,
10710     useDisplay: false
10711 });
10712 </code></pre>
10713     * @param {Object} options (optional) Object literal with any of the Fx config options
10714     * @return {Roo.Element} The Element
10715     */
10716     fadeOut : function(o){
10717         var el = this.getFxEl();
10718         o = o || {};
10719         el.queueFx(o, function(){
10720             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10721                 o, null, .5, "easeOut", function(){
10722                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10723                      this.dom.style.display = "none";
10724                 }else{
10725                      this.dom.style.visibility = "hidden";
10726                 }
10727                 this.clearOpacity();
10728                 el.afterFx(o);
10729             });
10730         });
10731         return this;
10732     },
10733
10734    /**
10735     * Animates the transition of an element's dimensions from a starting height/width
10736     * to an ending height/width.
10737     * Usage:
10738 <pre><code>
10739 // change height and width to 100x100 pixels
10740 el.scale(100, 100);
10741
10742 // common config options shown with default values.  The height and width will default to
10743 // the element's existing values if passed as null.
10744 el.scale(
10745     [element's width],
10746     [element's height], {
10747     easing: 'easeOut',
10748     duration: .35
10749 });
10750 </code></pre>
10751     * @param {Number} width  The new width (pass undefined to keep the original width)
10752     * @param {Number} height  The new height (pass undefined to keep the original height)
10753     * @param {Object} options (optional) Object literal with any of the Fx config options
10754     * @return {Roo.Element} The Element
10755     */
10756     scale : function(w, h, o){
10757         this.shift(Roo.apply({}, o, {
10758             width: w,
10759             height: h
10760         }));
10761         return this;
10762     },
10763
10764    /**
10765     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10766     * Any of these properties not specified in the config object will not be changed.  This effect 
10767     * requires that at least one new dimension, position or opacity setting must be passed in on
10768     * the config object in order for the function to have any effect.
10769     * Usage:
10770 <pre><code>
10771 // slide the element horizontally to x position 200 while changing the height and opacity
10772 el.shift({ x: 200, height: 50, opacity: .8 });
10773
10774 // common config options shown with default values.
10775 el.shift({
10776     width: [element's width],
10777     height: [element's height],
10778     x: [element's x position],
10779     y: [element's y position],
10780     opacity: [element's opacity],
10781     easing: 'easeOut',
10782     duration: .35
10783 });
10784 </code></pre>
10785     * @param {Object} options  Object literal with any of the Fx config options
10786     * @return {Roo.Element} The Element
10787     */
10788     shift : function(o){
10789         var el = this.getFxEl();
10790         o = o || {};
10791         el.queueFx(o, function(){
10792             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10793             if(w !== undefined){
10794                 a.width = {to: this.adjustWidth(w)};
10795             }
10796             if(h !== undefined){
10797                 a.height = {to: this.adjustHeight(h)};
10798             }
10799             if(x !== undefined || y !== undefined){
10800                 a.points = {to: [
10801                     x !== undefined ? x : this.getX(),
10802                     y !== undefined ? y : this.getY()
10803                 ]};
10804             }
10805             if(op !== undefined){
10806                 a.opacity = {to: op};
10807             }
10808             if(o.xy !== undefined){
10809                 a.points = {to: o.xy};
10810             }
10811             arguments.callee.anim = this.fxanim(a,
10812                 o, 'motion', .35, "easeOut", function(){
10813                 el.afterFx(o);
10814             });
10815         });
10816         return this;
10817     },
10818
10819         /**
10820          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10821          * ending point of the effect.
10822          * Usage:
10823          *<pre><code>
10824 // default: slide the element downward while fading out
10825 el.ghost();
10826
10827 // custom: slide the element out to the right with a 2-second duration
10828 el.ghost('r', { duration: 2 });
10829
10830 // common config options shown with default values
10831 el.ghost('b', {
10832     easing: 'easeOut',
10833     duration: .5
10834     remove: false,
10835     useDisplay: false
10836 });
10837 </code></pre>
10838          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10839          * @param {Object} options (optional) Object literal with any of the Fx config options
10840          * @return {Roo.Element} The Element
10841          */
10842     ghost : function(anchor, o){
10843         var el = this.getFxEl();
10844         o = o || {};
10845
10846         el.queueFx(o, function(){
10847             anchor = anchor || "b";
10848
10849             // restore values after effect
10850             var r = this.getFxRestore();
10851             var w = this.getWidth(),
10852                 h = this.getHeight();
10853
10854             var st = this.dom.style;
10855
10856             var after = function(){
10857                 if(o.useDisplay){
10858                     el.setDisplayed(false);
10859                 }else{
10860                     el.hide();
10861                 }
10862
10863                 el.clearOpacity();
10864                 el.setPositioning(r.pos);
10865                 st.width = r.width;
10866                 st.height = r.height;
10867
10868                 el.afterFx(o);
10869             };
10870
10871             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10872             switch(anchor.toLowerCase()){
10873                 case "t":
10874                     pt.by = [0, -h];
10875                 break;
10876                 case "l":
10877                     pt.by = [-w, 0];
10878                 break;
10879                 case "r":
10880                     pt.by = [w, 0];
10881                 break;
10882                 case "b":
10883                     pt.by = [0, h];
10884                 break;
10885                 case "tl":
10886                     pt.by = [-w, -h];
10887                 break;
10888                 case "bl":
10889                     pt.by = [-w, h];
10890                 break;
10891                 case "br":
10892                     pt.by = [w, h];
10893                 break;
10894                 case "tr":
10895                     pt.by = [w, -h];
10896                 break;
10897             }
10898
10899             arguments.callee.anim = this.fxanim(a,
10900                 o,
10901                 'motion',
10902                 .5,
10903                 "easeOut", after);
10904         });
10905         return this;
10906     },
10907
10908         /**
10909          * Ensures that all effects queued after syncFx is called on the element are
10910          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10911          * @return {Roo.Element} The Element
10912          */
10913     syncFx : function(){
10914         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10915             block : false,
10916             concurrent : true,
10917             stopFx : false
10918         });
10919         return this;
10920     },
10921
10922         /**
10923          * Ensures that all effects queued after sequenceFx is called on the element are
10924          * run in sequence.  This is the opposite of {@link #syncFx}.
10925          * @return {Roo.Element} The Element
10926          */
10927     sequenceFx : function(){
10928         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10929             block : false,
10930             concurrent : false,
10931             stopFx : false
10932         });
10933         return this;
10934     },
10935
10936         /* @private */
10937     nextFx : function(){
10938         var ef = this.fxQueue[0];
10939         if(ef){
10940             ef.call(this);
10941         }
10942     },
10943
10944         /**
10945          * Returns true if the element has any effects actively running or queued, else returns false.
10946          * @return {Boolean} True if element has active effects, else false
10947          */
10948     hasActiveFx : function(){
10949         return this.fxQueue && this.fxQueue[0];
10950     },
10951
10952         /**
10953          * Stops any running effects and clears the element's internal effects queue if it contains
10954          * any additional effects that haven't started yet.
10955          * @return {Roo.Element} The Element
10956          */
10957     stopFx : function(){
10958         if(this.hasActiveFx()){
10959             var cur = this.fxQueue[0];
10960             if(cur && cur.anim && cur.anim.isAnimated()){
10961                 this.fxQueue = [cur]; // clear out others
10962                 cur.anim.stop(true);
10963             }
10964         }
10965         return this;
10966     },
10967
10968         /* @private */
10969     beforeFx : function(o){
10970         if(this.hasActiveFx() && !o.concurrent){
10971            if(o.stopFx){
10972                this.stopFx();
10973                return true;
10974            }
10975            return false;
10976         }
10977         return true;
10978     },
10979
10980         /**
10981          * Returns true if the element is currently blocking so that no other effect can be queued
10982          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10983          * used to ensure that an effect initiated by a user action runs to completion prior to the
10984          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10985          * @return {Boolean} True if blocking, else false
10986          */
10987     hasFxBlock : function(){
10988         var q = this.fxQueue;
10989         return q && q[0] && q[0].block;
10990     },
10991
10992         /* @private */
10993     queueFx : function(o, fn){
10994         if(!this.fxQueue){
10995             this.fxQueue = [];
10996         }
10997         if(!this.hasFxBlock()){
10998             Roo.applyIf(o, this.fxDefaults);
10999             if(!o.concurrent){
11000                 var run = this.beforeFx(o);
11001                 fn.block = o.block;
11002                 this.fxQueue.push(fn);
11003                 if(run){
11004                     this.nextFx();
11005                 }
11006             }else{
11007                 fn.call(this);
11008             }
11009         }
11010         return this;
11011     },
11012
11013         /* @private */
11014     fxWrap : function(pos, o, vis){
11015         var wrap;
11016         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11017             var wrapXY;
11018             if(o.fixPosition){
11019                 wrapXY = this.getXY();
11020             }
11021             var div = document.createElement("div");
11022             div.style.visibility = vis;
11023             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11024             wrap.setPositioning(pos);
11025             if(wrap.getStyle("position") == "static"){
11026                 wrap.position("relative");
11027             }
11028             this.clearPositioning('auto');
11029             wrap.clip();
11030             wrap.dom.appendChild(this.dom);
11031             if(wrapXY){
11032                 wrap.setXY(wrapXY);
11033             }
11034         }
11035         return wrap;
11036     },
11037
11038         /* @private */
11039     fxUnwrap : function(wrap, pos, o){
11040         this.clearPositioning();
11041         this.setPositioning(pos);
11042         if(!o.wrap){
11043             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11044             wrap.remove();
11045         }
11046     },
11047
11048         /* @private */
11049     getFxRestore : function(){
11050         var st = this.dom.style;
11051         return {pos: this.getPositioning(), width: st.width, height : st.height};
11052     },
11053
11054         /* @private */
11055     afterFx : function(o){
11056         if(o.afterStyle){
11057             this.applyStyles(o.afterStyle);
11058         }
11059         if(o.afterCls){
11060             this.addClass(o.afterCls);
11061         }
11062         if(o.remove === true){
11063             this.remove();
11064         }
11065         Roo.callback(o.callback, o.scope, [this]);
11066         if(!o.concurrent){
11067             this.fxQueue.shift();
11068             this.nextFx();
11069         }
11070     },
11071
11072         /* @private */
11073     getFxEl : function(){ // support for composite element fx
11074         return Roo.get(this.dom);
11075     },
11076
11077         /* @private */
11078     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11079         animType = animType || 'run';
11080         opt = opt || {};
11081         var anim = Roo.lib.Anim[animType](
11082             this.dom, args,
11083             (opt.duration || defaultDur) || .35,
11084             (opt.easing || defaultEase) || 'easeOut',
11085             function(){
11086                 Roo.callback(cb, this);
11087             },
11088             this
11089         );
11090         opt.anim = anim;
11091         return anim;
11092     }
11093 };
11094
11095 // backwords compat
11096 Roo.Fx.resize = Roo.Fx.scale;
11097
11098 //When included, Roo.Fx is automatically applied to Element so that all basic
11099 //effects are available directly via the Element API
11100 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11101  * Based on:
11102  * Ext JS Library 1.1.1
11103  * Copyright(c) 2006-2007, Ext JS, LLC.
11104  *
11105  * Originally Released Under LGPL - original licence link has changed is not relivant.
11106  *
11107  * Fork - LGPL
11108  * <script type="text/javascript">
11109  */
11110
11111
11112 /**
11113  * @class Roo.CompositeElement
11114  * Standard composite class. Creates a Roo.Element for every element in the collection.
11115  * <br><br>
11116  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11117  * actions will be performed on all the elements in this collection.</b>
11118  * <br><br>
11119  * All methods return <i>this</i> and can be chained.
11120  <pre><code>
11121  var els = Roo.select("#some-el div.some-class", true);
11122  // or select directly from an existing element
11123  var el = Roo.get('some-el');
11124  el.select('div.some-class', true);
11125
11126  els.setWidth(100); // all elements become 100 width
11127  els.hide(true); // all elements fade out and hide
11128  // or
11129  els.setWidth(100).hide(true);
11130  </code></pre>
11131  */
11132 Roo.CompositeElement = function(els){
11133     this.elements = [];
11134     this.addElements(els);
11135 };
11136 Roo.CompositeElement.prototype = {
11137     isComposite: true,
11138     addElements : function(els){
11139         if(!els) {
11140             return this;
11141         }
11142         if(typeof els == "string"){
11143             els = Roo.Element.selectorFunction(els);
11144         }
11145         var yels = this.elements;
11146         var index = yels.length-1;
11147         for(var i = 0, len = els.length; i < len; i++) {
11148                 yels[++index] = Roo.get(els[i]);
11149         }
11150         return this;
11151     },
11152
11153     /**
11154     * Clears this composite and adds the elements returned by the passed selector.
11155     * @param {String/Array} els A string CSS selector, an array of elements or an element
11156     * @return {CompositeElement} this
11157     */
11158     fill : function(els){
11159         this.elements = [];
11160         this.add(els);
11161         return this;
11162     },
11163
11164     /**
11165     * Filters this composite to only elements that match the passed selector.
11166     * @param {String} selector A string CSS selector
11167     * @param {Boolean} inverse return inverse filter (not matches)
11168     * @return {CompositeElement} this
11169     */
11170     filter : function(selector, inverse){
11171         var els = [];
11172         inverse = inverse || false;
11173         this.each(function(el){
11174             var match = inverse ? !el.is(selector) : el.is(selector);
11175             if(match){
11176                 els[els.length] = el.dom;
11177             }
11178         });
11179         this.fill(els);
11180         return this;
11181     },
11182
11183     invoke : function(fn, args){
11184         var els = this.elements;
11185         for(var i = 0, len = els.length; i < len; i++) {
11186                 Roo.Element.prototype[fn].apply(els[i], args);
11187         }
11188         return this;
11189     },
11190     /**
11191     * Adds elements to this composite.
11192     * @param {String/Array} els A string CSS selector, an array of elements or an element
11193     * @return {CompositeElement} this
11194     */
11195     add : function(els){
11196         if(typeof els == "string"){
11197             this.addElements(Roo.Element.selectorFunction(els));
11198         }else if(els.length !== undefined){
11199             this.addElements(els);
11200         }else{
11201             this.addElements([els]);
11202         }
11203         return this;
11204     },
11205     /**
11206     * Calls the passed function passing (el, this, index) for each element in this composite.
11207     * @param {Function} fn The function to call
11208     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11209     * @return {CompositeElement} this
11210     */
11211     each : function(fn, scope){
11212         var els = this.elements;
11213         for(var i = 0, len = els.length; i < len; i++){
11214             if(fn.call(scope || els[i], els[i], this, i) === false) {
11215                 break;
11216             }
11217         }
11218         return this;
11219     },
11220
11221     /**
11222      * Returns the Element object at the specified index
11223      * @param {Number} index
11224      * @return {Roo.Element}
11225      */
11226     item : function(index){
11227         return this.elements[index] || null;
11228     },
11229
11230     /**
11231      * Returns the first Element
11232      * @return {Roo.Element}
11233      */
11234     first : function(){
11235         return this.item(0);
11236     },
11237
11238     /**
11239      * Returns the last Element
11240      * @return {Roo.Element}
11241      */
11242     last : function(){
11243         return this.item(this.elements.length-1);
11244     },
11245
11246     /**
11247      * Returns the number of elements in this composite
11248      * @return Number
11249      */
11250     getCount : function(){
11251         return this.elements.length;
11252     },
11253
11254     /**
11255      * Returns true if this composite contains the passed element
11256      * @return Boolean
11257      */
11258     contains : function(el){
11259         return this.indexOf(el) !== -1;
11260     },
11261
11262     /**
11263      * Returns true if this composite contains the passed element
11264      * @return Boolean
11265      */
11266     indexOf : function(el){
11267         return this.elements.indexOf(Roo.get(el));
11268     },
11269
11270
11271     /**
11272     * Removes the specified element(s).
11273     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11274     * or an array of any of those.
11275     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11276     * @return {CompositeElement} this
11277     */
11278     removeElement : function(el, removeDom){
11279         if(el instanceof Array){
11280             for(var i = 0, len = el.length; i < len; i++){
11281                 this.removeElement(el[i]);
11282             }
11283             return this;
11284         }
11285         var index = typeof el == 'number' ? el : this.indexOf(el);
11286         if(index !== -1){
11287             if(removeDom){
11288                 var d = this.elements[index];
11289                 if(d.dom){
11290                     d.remove();
11291                 }else{
11292                     d.parentNode.removeChild(d);
11293                 }
11294             }
11295             this.elements.splice(index, 1);
11296         }
11297         return this;
11298     },
11299
11300     /**
11301     * Replaces the specified element with the passed element.
11302     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11303     * to replace.
11304     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11305     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11306     * @return {CompositeElement} this
11307     */
11308     replaceElement : function(el, replacement, domReplace){
11309         var index = typeof el == 'number' ? el : this.indexOf(el);
11310         if(index !== -1){
11311             if(domReplace){
11312                 this.elements[index].replaceWith(replacement);
11313             }else{
11314                 this.elements.splice(index, 1, Roo.get(replacement))
11315             }
11316         }
11317         return this;
11318     },
11319
11320     /**
11321      * Removes all elements.
11322      */
11323     clear : function(){
11324         this.elements = [];
11325     }
11326 };
11327 (function(){
11328     Roo.CompositeElement.createCall = function(proto, fnName){
11329         if(!proto[fnName]){
11330             proto[fnName] = function(){
11331                 return this.invoke(fnName, arguments);
11332             };
11333         }
11334     };
11335     for(var fnName in Roo.Element.prototype){
11336         if(typeof Roo.Element.prototype[fnName] == "function"){
11337             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11338         }
11339     };
11340 })();
11341 /*
11342  * Based on:
11343  * Ext JS Library 1.1.1
11344  * Copyright(c) 2006-2007, Ext JS, LLC.
11345  *
11346  * Originally Released Under LGPL - original licence link has changed is not relivant.
11347  *
11348  * Fork - LGPL
11349  * <script type="text/javascript">
11350  */
11351
11352 /**
11353  * @class Roo.CompositeElementLite
11354  * @extends Roo.CompositeElement
11355  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11356  <pre><code>
11357  var els = Roo.select("#some-el div.some-class");
11358  // or select directly from an existing element
11359  var el = Roo.get('some-el');
11360  el.select('div.some-class');
11361
11362  els.setWidth(100); // all elements become 100 width
11363  els.hide(true); // all elements fade out and hide
11364  // or
11365  els.setWidth(100).hide(true);
11366  </code></pre><br><br>
11367  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11368  * actions will be performed on all the elements in this collection.</b>
11369  */
11370 Roo.CompositeElementLite = function(els){
11371     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11372     this.el = new Roo.Element.Flyweight();
11373 };
11374 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11375     addElements : function(els){
11376         if(els){
11377             if(els instanceof Array){
11378                 this.elements = this.elements.concat(els);
11379             }else{
11380                 var yels = this.elements;
11381                 var index = yels.length-1;
11382                 for(var i = 0, len = els.length; i < len; i++) {
11383                     yels[++index] = els[i];
11384                 }
11385             }
11386         }
11387         return this;
11388     },
11389     invoke : function(fn, args){
11390         var els = this.elements;
11391         var el = this.el;
11392         for(var i = 0, len = els.length; i < len; i++) {
11393             el.dom = els[i];
11394                 Roo.Element.prototype[fn].apply(el, args);
11395         }
11396         return this;
11397     },
11398     /**
11399      * Returns a flyweight Element of the dom element object at the specified index
11400      * @param {Number} index
11401      * @return {Roo.Element}
11402      */
11403     item : function(index){
11404         if(!this.elements[index]){
11405             return null;
11406         }
11407         this.el.dom = this.elements[index];
11408         return this.el;
11409     },
11410
11411     // fixes scope with flyweight
11412     addListener : function(eventName, handler, scope, opt){
11413         var els = this.elements;
11414         for(var i = 0, len = els.length; i < len; i++) {
11415             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11416         }
11417         return this;
11418     },
11419
11420     /**
11421     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11422     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11423     * a reference to the dom node, use el.dom.</b>
11424     * @param {Function} fn The function to call
11425     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11426     * @return {CompositeElement} this
11427     */
11428     each : function(fn, scope){
11429         var els = this.elements;
11430         var el = this.el;
11431         for(var i = 0, len = els.length; i < len; i++){
11432             el.dom = els[i];
11433                 if(fn.call(scope || el, el, this, i) === false){
11434                 break;
11435             }
11436         }
11437         return this;
11438     },
11439
11440     indexOf : function(el){
11441         return this.elements.indexOf(Roo.getDom(el));
11442     },
11443
11444     replaceElement : function(el, replacement, domReplace){
11445         var index = typeof el == 'number' ? el : this.indexOf(el);
11446         if(index !== -1){
11447             replacement = Roo.getDom(replacement);
11448             if(domReplace){
11449                 var d = this.elements[index];
11450                 d.parentNode.insertBefore(replacement, d);
11451                 d.parentNode.removeChild(d);
11452             }
11453             this.elements.splice(index, 1, replacement);
11454         }
11455         return this;
11456     }
11457 });
11458 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11459
11460 /*
11461  * Based on:
11462  * Ext JS Library 1.1.1
11463  * Copyright(c) 2006-2007, Ext JS, LLC.
11464  *
11465  * Originally Released Under LGPL - original licence link has changed is not relivant.
11466  *
11467  * Fork - LGPL
11468  * <script type="text/javascript">
11469  */
11470
11471  
11472
11473 /**
11474  * @class Roo.data.Connection
11475  * @extends Roo.util.Observable
11476  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11477  * either to a configured URL, or to a URL specified at request time. 
11478  * 
11479  * Requests made by this class are asynchronous, and will return immediately. No data from
11480  * the server will be available to the statement immediately following the {@link #request} call.
11481  * To process returned data, use a callback in the request options object, or an event listener.
11482  * 
11483  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11484  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11485  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11486  * property and, if present, the IFRAME's XML document as the responseXML property.
11487  * 
11488  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11489  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11490  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11491  * standard DOM methods.
11492  * @constructor
11493  * @param {Object} config a configuration object.
11494  */
11495 Roo.data.Connection = function(config){
11496     Roo.apply(this, config);
11497     this.addEvents({
11498         /**
11499          * @event beforerequest
11500          * Fires before a network request is made to retrieve a data object.
11501          * @param {Connection} conn This Connection object.
11502          * @param {Object} options The options config object passed to the {@link #request} method.
11503          */
11504         "beforerequest" : true,
11505         /**
11506          * @event requestcomplete
11507          * Fires if the request was successfully completed.
11508          * @param {Connection} conn This Connection object.
11509          * @param {Object} response The XHR object containing the response data.
11510          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11511          * @param {Object} options The options config object passed to the {@link #request} method.
11512          */
11513         "requestcomplete" : true,
11514         /**
11515          * @event requestexception
11516          * Fires if an error HTTP status was returned from the server.
11517          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11518          * @param {Connection} conn This Connection object.
11519          * @param {Object} response The XHR object containing the response data.
11520          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11521          * @param {Object} options The options config object passed to the {@link #request} method.
11522          */
11523         "requestexception" : true
11524     });
11525     Roo.data.Connection.superclass.constructor.call(this);
11526 };
11527
11528 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11529     /**
11530      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11531      */
11532     /**
11533      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11534      * extra parameters to each request made by this object. (defaults to undefined)
11535      */
11536     /**
11537      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11538      *  to each request made by this object. (defaults to undefined)
11539      */
11540     /**
11541      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11542      */
11543     /**
11544      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11545      */
11546     timeout : 30000,
11547     /**
11548      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11549      * @type Boolean
11550      */
11551     autoAbort:false,
11552
11553     /**
11554      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11555      * @type Boolean
11556      */
11557     disableCaching: true,
11558
11559     /**
11560      * Sends an HTTP request to a remote server.
11561      * @param {Object} options An object which may contain the following properties:<ul>
11562      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11563      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11564      * request, a url encoded string or a function to call to get either.</li>
11565      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11566      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11567      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11568      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11569      * <li>options {Object} The parameter to the request call.</li>
11570      * <li>success {Boolean} True if the request succeeded.</li>
11571      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11572      * </ul></li>
11573      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11574      * The callback is passed the following parameters:<ul>
11575      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11576      * <li>options {Object} The parameter to the request call.</li>
11577      * </ul></li>
11578      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11579      * The callback is passed the following parameters:<ul>
11580      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11581      * <li>options {Object} The parameter to the request call.</li>
11582      * </ul></li>
11583      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11584      * for the callback function. Defaults to the browser window.</li>
11585      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11586      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11587      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11588      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11589      * params for the post data. Any params will be appended to the URL.</li>
11590      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11591      * </ul>
11592      * @return {Number} transactionId
11593      */
11594     request : function(o){
11595         if(this.fireEvent("beforerequest", this, o) !== false){
11596             var p = o.params;
11597
11598             if(typeof p == "function"){
11599                 p = p.call(o.scope||window, o);
11600             }
11601             if(typeof p == "object"){
11602                 p = Roo.urlEncode(o.params);
11603             }
11604             if(this.extraParams){
11605                 var extras = Roo.urlEncode(this.extraParams);
11606                 p = p ? (p + '&' + extras) : extras;
11607             }
11608
11609             var url = o.url || this.url;
11610             if(typeof url == 'function'){
11611                 url = url.call(o.scope||window, o);
11612             }
11613
11614             if(o.form){
11615                 var form = Roo.getDom(o.form);
11616                 url = url || form.action;
11617
11618                 var enctype = form.getAttribute("enctype");
11619                 
11620                 if (o.formData) {
11621                     return this.doFormDataUpload(o,p,url);
11622                 }
11623                 
11624                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11625                     return this.doFormUpload(o, p, url);
11626                 }
11627                 var f = Roo.lib.Ajax.serializeForm(form);
11628                 p = p ? (p + '&' + f) : f;
11629             }
11630
11631             var hs = o.headers;
11632             if(this.defaultHeaders){
11633                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11634                 if(!o.headers){
11635                     o.headers = hs;
11636                 }
11637             }
11638
11639             var cb = {
11640                 success: this.handleResponse,
11641                 failure: this.handleFailure,
11642                 scope: this,
11643                 argument: {options: o},
11644                 timeout : o.timeout || this.timeout
11645             };
11646
11647             var method = o.method||this.method||(p ? "POST" : "GET");
11648
11649             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11650                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11651             }
11652
11653             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11654                 if(o.autoAbort){
11655                     this.abort();
11656                 }
11657             }else if(this.autoAbort !== false){
11658                 this.abort();
11659             }
11660
11661             if((method == 'GET' && p) || o.xmlData){
11662                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11663                 p = '';
11664             }
11665             Roo.lib.Ajax.useDefaultHeader = typeof(o.headers) == 'undefined' || typeof(o.headers['Content-Type']) == 'undefined';
11666             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11667             Roo.lib.Ajax.useDefaultHeader == true;
11668             return this.transId;
11669         }else{
11670             Roo.callback(o.callback, o.scope, [o, null, null]);
11671             return null;
11672         }
11673     },
11674
11675     /**
11676      * Determine whether this object has a request outstanding.
11677      * @param {Number} transactionId (Optional) defaults to the last transaction
11678      * @return {Boolean} True if there is an outstanding request.
11679      */
11680     isLoading : function(transId){
11681         if(transId){
11682             return Roo.lib.Ajax.isCallInProgress(transId);
11683         }else{
11684             return this.transId ? true : false;
11685         }
11686     },
11687
11688     /**
11689      * Aborts any outstanding request.
11690      * @param {Number} transactionId (Optional) defaults to the last transaction
11691      */
11692     abort : function(transId){
11693         if(transId || this.isLoading()){
11694             Roo.lib.Ajax.abort(transId || this.transId);
11695         }
11696     },
11697
11698     // private
11699     handleResponse : function(response){
11700         this.transId = false;
11701         var options = response.argument.options;
11702         response.argument = options ? options.argument : null;
11703         this.fireEvent("requestcomplete", this, response, options);
11704         Roo.callback(options.success, options.scope, [response, options]);
11705         Roo.callback(options.callback, options.scope, [options, true, response]);
11706     },
11707
11708     // private
11709     handleFailure : function(response, e){
11710         this.transId = false;
11711         var options = response.argument.options;
11712         response.argument = options ? options.argument : null;
11713         this.fireEvent("requestexception", this, response, options, e);
11714         Roo.callback(options.failure, options.scope, [response, options]);
11715         Roo.callback(options.callback, options.scope, [options, false, response]);
11716     },
11717
11718     // private
11719     doFormUpload : function(o, ps, url){
11720         var id = Roo.id();
11721         var frame = document.createElement('iframe');
11722         frame.id = id;
11723         frame.name = id;
11724         frame.className = 'x-hidden';
11725         if(Roo.isIE){
11726             frame.src = Roo.SSL_SECURE_URL;
11727         }
11728         document.body.appendChild(frame);
11729
11730         if(Roo.isIE){
11731            document.frames[id].name = id;
11732         }
11733
11734         var form = Roo.getDom(o.form);
11735         form.target = id;
11736         form.method = 'POST';
11737         form.enctype = form.encoding = 'multipart/form-data';
11738         if(url){
11739             form.action = url;
11740         }
11741
11742         var hiddens, hd;
11743         if(ps){ // add dynamic params
11744             hiddens = [];
11745             ps = Roo.urlDecode(ps, false);
11746             for(var k in ps){
11747                 if(ps.hasOwnProperty(k)){
11748                     hd = document.createElement('input');
11749                     hd.type = 'hidden';
11750                     hd.name = k;
11751                     hd.value = ps[k];
11752                     form.appendChild(hd);
11753                     hiddens.push(hd);
11754                 }
11755             }
11756         }
11757
11758         function cb(){
11759             var r = {  // bogus response object
11760                 responseText : '',
11761                 responseXML : null
11762             };
11763
11764             r.argument = o ? o.argument : null;
11765
11766             try { //
11767                 var doc;
11768                 if(Roo.isIE){
11769                     doc = frame.contentWindow.document;
11770                 }else {
11771                     doc = (frame.contentDocument || window.frames[id].document);
11772                 }
11773                 if(doc && doc.body){
11774                     r.responseText = doc.body.innerHTML;
11775                 }
11776                 if(doc && doc.XMLDocument){
11777                     r.responseXML = doc.XMLDocument;
11778                 }else {
11779                     r.responseXML = doc;
11780                 }
11781             }
11782             catch(e) {
11783                 // ignore
11784             }
11785
11786             Roo.EventManager.removeListener(frame, 'load', cb, this);
11787
11788             this.fireEvent("requestcomplete", this, r, o);
11789             Roo.callback(o.success, o.scope, [r, o]);
11790             Roo.callback(o.callback, o.scope, [o, true, r]);
11791
11792             setTimeout(function(){document.body.removeChild(frame);}, 100);
11793         }
11794
11795         Roo.EventManager.on(frame, 'load', cb, this);
11796         form.submit();
11797
11798         if(hiddens){ // remove dynamic params
11799             for(var i = 0, len = hiddens.length; i < len; i++){
11800                 form.removeChild(hiddens[i]);
11801             }
11802         }
11803     },
11804     // this is a 'formdata version???'
11805     
11806     
11807     doFormDataUpload : function(o, ps, url)
11808     {
11809         var form = Roo.getDom(o.form);
11810         form.enctype = form.encoding = 'multipart/form-data';
11811         var formData = o.formData === true ? new FormData(form) : o.formData;
11812       
11813         var cb = {
11814             success: this.handleResponse,
11815             failure: this.handleFailure,
11816             scope: this,
11817             argument: {options: o},
11818             timeout : o.timeout || this.timeout
11819         };
11820  
11821         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11822             if(o.autoAbort){
11823                 this.abort();
11824             }
11825         }else if(this.autoAbort !== false){
11826             this.abort();
11827         }
11828
11829         //Roo.lib.Ajax.defaultPostHeader = null;
11830         Roo.lib.Ajax.useDefaultHeader = false;
11831         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11832         Roo.lib.Ajax.useDefaultHeader = true;
11833  
11834          
11835     }
11836     
11837 });
11838 /*
11839  * Based on:
11840  * Ext JS Library 1.1.1
11841  * Copyright(c) 2006-2007, Ext JS, LLC.
11842  *
11843  * Originally Released Under LGPL - original licence link has changed is not relivant.
11844  *
11845  * Fork - LGPL
11846  * <script type="text/javascript">
11847  */
11848  
11849 /**
11850  * Global Ajax request class.
11851  * 
11852  * @class Roo.Ajax
11853  * @extends Roo.data.Connection
11854  * @static
11855  * 
11856  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11857  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11858  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11859  * @cfg {String} method (Optional)  The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11860  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11861  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11862  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11863  */
11864 Roo.Ajax = new Roo.data.Connection({
11865     // fix up the docs
11866     /**
11867      * @scope Roo.Ajax
11868      * @type {Boolear} 
11869      */
11870     autoAbort : false,
11871
11872     /**
11873      * Serialize the passed form into a url encoded string
11874      * @scope Roo.Ajax
11875      * @param {String/HTMLElement} form
11876      * @return {String}
11877      */
11878     serializeForm : function(form){
11879         return Roo.lib.Ajax.serializeForm(form);
11880     }
11881 });/*
11882  * Based on:
11883  * Ext JS Library 1.1.1
11884  * Copyright(c) 2006-2007, Ext JS, LLC.
11885  *
11886  * Originally Released Under LGPL - original licence link has changed is not relivant.
11887  *
11888  * Fork - LGPL
11889  * <script type="text/javascript">
11890  */
11891
11892  
11893 /**
11894  * @class Roo.UpdateManager
11895  * @extends Roo.util.Observable
11896  * Provides AJAX-style update for Element object.<br><br>
11897  * Usage:<br>
11898  * <pre><code>
11899  * // Get it from a Roo.Element object
11900  * var el = Roo.get("foo");
11901  * var mgr = el.getUpdateManager();
11902  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11903  * ...
11904  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11905  * <br>
11906  * // or directly (returns the same UpdateManager instance)
11907  * var mgr = new Roo.UpdateManager("myElementId");
11908  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11909  * mgr.on("update", myFcnNeedsToKnow);
11910  * <br>
11911    // short handed call directly from the element object
11912    Roo.get("foo").load({
11913         url: "bar.php",
11914         scripts:true,
11915         params: "for=bar",
11916         text: "Loading Foo..."
11917    });
11918  * </code></pre>
11919  * @constructor
11920  * Create new UpdateManager directly.
11921  * @param {String/HTMLElement/Roo.Element} el The element to update
11922  * @param {Boolean} forceNew (optional) By default the constructor checks to see if the passed element already has an UpdateManager and if it does it returns the same instance. This will skip that check (useful for extending this class).
11923  */
11924 Roo.UpdateManager = function(el, forceNew){
11925     el = Roo.get(el);
11926     if(!forceNew && el.updateManager){
11927         return el.updateManager;
11928     }
11929     /**
11930      * The Element object
11931      * @type Roo.Element
11932      */
11933     this.el = el;
11934     /**
11935      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11936      * @type String
11937      */
11938     this.defaultUrl = null;
11939
11940     this.addEvents({
11941         /**
11942          * @event beforeupdate
11943          * Fired before an update is made, return false from your handler and the update is cancelled.
11944          * @param {Roo.Element} el
11945          * @param {String/Object/Function} url
11946          * @param {String/Object} params
11947          */
11948         "beforeupdate": true,
11949         /**
11950          * @event update
11951          * Fired after successful update is made.
11952          * @param {Roo.Element} el
11953          * @param {Object} oResponseObject The response Object
11954          */
11955         "update": true,
11956         /**
11957          * @event failure
11958          * Fired on update failure.
11959          * @param {Roo.Element} el
11960          * @param {Object} oResponseObject The response Object
11961          */
11962         "failure": true
11963     });
11964     var d = Roo.UpdateManager.defaults;
11965     /**
11966      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11967      * @type String
11968      */
11969     this.sslBlankUrl = d.sslBlankUrl;
11970     /**
11971      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11972      * @type Boolean
11973      */
11974     this.disableCaching = d.disableCaching;
11975     /**
11976      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11977      * @type String
11978      */
11979     this.indicatorText = d.indicatorText;
11980     /**
11981      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11982      * @type String
11983      */
11984     this.showLoadIndicator = d.showLoadIndicator;
11985     /**
11986      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11987      * @type Number
11988      */
11989     this.timeout = d.timeout;
11990
11991     /**
11992      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11993      * @type Boolean
11994      */
11995     this.loadScripts = d.loadScripts;
11996
11997     /**
11998      * Transaction object of current executing transaction
11999      */
12000     this.transaction = null;
12001
12002     /**
12003      * @private
12004      */
12005     this.autoRefreshProcId = null;
12006     /**
12007      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12008      * @type Function
12009      */
12010     this.refreshDelegate = this.refresh.createDelegate(this);
12011     /**
12012      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12013      * @type Function
12014      */
12015     this.updateDelegate = this.update.createDelegate(this);
12016     /**
12017      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12018      * @type Function
12019      */
12020     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12021     /**
12022      * @private
12023      */
12024     this.successDelegate = this.processSuccess.createDelegate(this);
12025     /**
12026      * @private
12027      */
12028     this.failureDelegate = this.processFailure.createDelegate(this);
12029
12030     if(!this.renderer){
12031      /**
12032       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12033       */
12034     this.renderer = new Roo.UpdateManager.BasicRenderer();
12035     }
12036     
12037     Roo.UpdateManager.superclass.constructor.call(this);
12038 };
12039
12040 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12041     /**
12042      * Get the Element this UpdateManager is bound to
12043      * @return {Roo.Element} The element
12044      */
12045     getEl : function(){
12046         return this.el;
12047     },
12048     /**
12049      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12050      * @param {Object/String/Function} url The url for this request or a function to call to get the url or a config object containing any of the following options:
12051 <pre><code>
12052 um.update({<br/>
12053     url: "your-url.php",<br/>
12054     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12055     callback: yourFunction,<br/>
12056     scope: yourObject, //(optional scope)  <br/>
12057     discardUrl: false, <br/>
12058     nocache: false,<br/>
12059     text: "Loading...",<br/>
12060     timeout: 30,<br/>
12061     scripts: false<br/>
12062 });
12063 </code></pre>
12064      * The only required property is url. The optional properties nocache, text and scripts
12065      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12066      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
12067      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12068      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
12069      */
12070     update : function(url, params, callback, discardUrl){
12071         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12072             var method = this.method,
12073                 cfg;
12074             if(typeof url == "object"){ // must be config object
12075                 cfg = url;
12076                 url = cfg.url;
12077                 params = params || cfg.params;
12078                 callback = callback || cfg.callback;
12079                 discardUrl = discardUrl || cfg.discardUrl;
12080                 if(callback && cfg.scope){
12081                     callback = callback.createDelegate(cfg.scope);
12082                 }
12083                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12084                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12085                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12086                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12087                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12088             }
12089             this.showLoading();
12090             if(!discardUrl){
12091                 this.defaultUrl = url;
12092             }
12093             if(typeof url == "function"){
12094                 url = url.call(this);
12095             }
12096
12097             method = method || (params ? "POST" : "GET");
12098             if(method == "GET"){
12099                 url = this.prepareUrl(url);
12100             }
12101
12102             var o = Roo.apply(cfg ||{}, {
12103                 url : url,
12104                 params: params,
12105                 success: this.successDelegate,
12106                 failure: this.failureDelegate,
12107                 callback: undefined,
12108                 timeout: (this.timeout*1000),
12109                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12110             });
12111             Roo.log("updated manager called with timeout of " + o.timeout);
12112             this.transaction = Roo.Ajax.request(o);
12113         }
12114     },
12115
12116     /**
12117      * Performs an async form post, updating this element with the response. If the form has the attribute enctype="multipart/form-data", it assumes it's a file upload.
12118      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12119      * @param {String/HTMLElement} form The form Id or form element
12120      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12121      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12122      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12123      */
12124     formUpdate : function(form, url, reset, callback){
12125         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12126             if(typeof url == "function"){
12127                 url = url.call(this);
12128             }
12129             form = Roo.getDom(form);
12130             this.transaction = Roo.Ajax.request({
12131                 form: form,
12132                 url:url,
12133                 success: this.successDelegate,
12134                 failure: this.failureDelegate,
12135                 timeout: (this.timeout*1000),
12136                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12137             });
12138             this.showLoading.defer(1, this);
12139         }
12140     },
12141
12142     /**
12143      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12144      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12145      */
12146     refresh : function(callback){
12147         if(this.defaultUrl == null){
12148             return;
12149         }
12150         this.update(this.defaultUrl, null, callback, true);
12151     },
12152
12153     /**
12154      * Set this element to auto refresh.
12155      * @param {Number} interval How often to update (in seconds).
12156      * @param {String/Function} url (optional) The url for this request or a function to call to get the url (Defaults to the last used url)
12157      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "&param1=1&param2=2" or as an object {param1: 1, param2: 2}
12158      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12159      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12160      */
12161     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12162         if(refreshNow){
12163             this.update(url || this.defaultUrl, params, callback, true);
12164         }
12165         if(this.autoRefreshProcId){
12166             clearInterval(this.autoRefreshProcId);
12167         }
12168         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12169     },
12170
12171     /**
12172      * Stop auto refresh on this element.
12173      */
12174      stopAutoRefresh : function(){
12175         if(this.autoRefreshProcId){
12176             clearInterval(this.autoRefreshProcId);
12177             delete this.autoRefreshProcId;
12178         }
12179     },
12180
12181     isAutoRefreshing : function(){
12182        return this.autoRefreshProcId ? true : false;
12183     },
12184     /**
12185      * Called to update the element to "Loading" state. Override to perform custom action.
12186      */
12187     showLoading : function(){
12188         if(this.showLoadIndicator){
12189             this.el.update(this.indicatorText);
12190         }
12191     },
12192
12193     /**
12194      * Adds unique parameter to query string if disableCaching = true
12195      * @private
12196      */
12197     prepareUrl : function(url){
12198         if(this.disableCaching){
12199             var append = "_dc=" + (new Date().getTime());
12200             if(url.indexOf("?") !== -1){
12201                 url += "&" + append;
12202             }else{
12203                 url += "?" + append;
12204             }
12205         }
12206         return url;
12207     },
12208
12209     /**
12210      * @private
12211      */
12212     processSuccess : function(response){
12213         this.transaction = null;
12214         if(response.argument.form && response.argument.reset){
12215             try{ // put in try/catch since some older FF releases had problems with this
12216                 response.argument.form.reset();
12217             }catch(e){}
12218         }
12219         if(this.loadScripts){
12220             this.renderer.render(this.el, response, this,
12221                 this.updateComplete.createDelegate(this, [response]));
12222         }else{
12223             this.renderer.render(this.el, response, this);
12224             this.updateComplete(response);
12225         }
12226     },
12227
12228     updateComplete : function(response){
12229         this.fireEvent("update", this.el, response);
12230         if(typeof response.argument.callback == "function"){
12231             response.argument.callback(this.el, true, response);
12232         }
12233     },
12234
12235     /**
12236      * @private
12237      */
12238     processFailure : function(response){
12239         this.transaction = null;
12240         this.fireEvent("failure", this.el, response);
12241         if(typeof response.argument.callback == "function"){
12242             response.argument.callback(this.el, false, response);
12243         }
12244     },
12245
12246     /**
12247      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12248      * @param {Object} renderer The object implementing the render() method
12249      */
12250     setRenderer : function(renderer){
12251         this.renderer = renderer;
12252     },
12253
12254     getRenderer : function(){
12255        return this.renderer;
12256     },
12257
12258     /**
12259      * Set the defaultUrl used for updates
12260      * @param {String/Function} defaultUrl The url or a function to call to get the url
12261      */
12262     setDefaultUrl : function(defaultUrl){
12263         this.defaultUrl = defaultUrl;
12264     },
12265
12266     /**
12267      * Aborts the executing transaction
12268      */
12269     abort : function(){
12270         if(this.transaction){
12271             Roo.Ajax.abort(this.transaction);
12272         }
12273     },
12274
12275     /**
12276      * Returns true if an update is in progress
12277      * @return {Boolean}
12278      */
12279     isUpdating : function(){
12280         if(this.transaction){
12281             return Roo.Ajax.isLoading(this.transaction);
12282         }
12283         return false;
12284     }
12285 });
12286
12287 /**
12288  * @class Roo.UpdateManager.defaults
12289  * @static (not really - but it helps the doc tool)
12290  * The defaults collection enables customizing the default properties of UpdateManager
12291  */
12292    Roo.UpdateManager.defaults = {
12293        /**
12294          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12295          * @type Number
12296          */
12297          timeout : 30,
12298
12299          /**
12300          * True to process scripts by default (Defaults to false).
12301          * @type Boolean
12302          */
12303         loadScripts : false,
12304
12305         /**
12306         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12307         * @type String
12308         */
12309         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12310         /**
12311          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12312          * @type Boolean
12313          */
12314         disableCaching : false,
12315         /**
12316          * Whether to show indicatorText when loading (Defaults to true).
12317          * @type Boolean
12318          */
12319         showLoadIndicator : true,
12320         /**
12321          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12322          * @type String
12323          */
12324         indicatorText : '<div class="loading-indicator">Loading...</div>'
12325    };
12326
12327 /**
12328  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12329  *Usage:
12330  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12331  * @param {String/HTMLElement/Roo.Element} el The element to update
12332  * @param {String} url The url
12333  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12334  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12335  * @static
12336  * @deprecated
12337  * @member Roo.UpdateManager
12338  */
12339 Roo.UpdateManager.updateElement = function(el, url, params, options){
12340     var um = Roo.get(el, true).getUpdateManager();
12341     Roo.apply(um, options);
12342     um.update(url, params, options ? options.callback : null);
12343 };
12344 // alias for backwards compat
12345 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12346 /**
12347  * @class Roo.UpdateManager.BasicRenderer
12348  * Default Content renderer. Updates the elements innerHTML with the responseText.
12349  */
12350 Roo.UpdateManager.BasicRenderer = function(){};
12351
12352 Roo.UpdateManager.BasicRenderer.prototype = {
12353     /**
12354      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12355      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12356      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12357      * @param {Roo.Element} el The element being rendered
12358      * @param {Object} response The YUI Connect response object
12359      * @param {UpdateManager} updateManager The calling update manager
12360      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12361      */
12362      render : function(el, response, updateManager, callback){
12363         el.update(response.responseText, updateManager.loadScripts, callback);
12364     }
12365 };
12366 /*
12367  * Based on:
12368  * Roo JS
12369  * (c)) Alan Knowles
12370  * Licence : LGPL
12371  */
12372
12373
12374 /**
12375  * @class Roo.DomTemplate
12376  * @extends Roo.Template
12377  * An effort at a dom based template engine..
12378  *
12379  * Similar to XTemplate, except it uses dom parsing to create the template..
12380  *
12381  * Supported features:
12382  *
12383  *  Tags:
12384
12385 <pre><code>
12386       {a_variable} - output encoded.
12387       {a_variable.format:("Y-m-d")} - call a method on the variable
12388       {a_variable:raw} - unencoded output
12389       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12390       {a_variable:this.method_on_template(...)} - call a method on the template object.
12391  
12392 </code></pre>
12393  *  The tpl tag:
12394 <pre><code>
12395         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12396         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12397         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12398         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12399   
12400 </code></pre>
12401  *      
12402  */
12403 Roo.DomTemplate = function()
12404 {
12405      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12406      if (this.html) {
12407         this.compile();
12408      }
12409 };
12410
12411
12412 Roo.extend(Roo.DomTemplate, Roo.Template, {
12413     /**
12414      * id counter for sub templates.
12415      */
12416     id : 0,
12417     /**
12418      * flag to indicate if dom parser is inside a pre,
12419      * it will strip whitespace if not.
12420      */
12421     inPre : false,
12422     
12423     /**
12424      * The various sub templates
12425      */
12426     tpls : false,
12427     
12428     
12429     
12430     /**
12431      *
12432      * basic tag replacing syntax
12433      * WORD:WORD()
12434      *
12435      * // you can fake an object call by doing this
12436      *  x.t:(test,tesT) 
12437      * 
12438      */
12439     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12440     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12441     
12442     iterChild : function (node, method) {
12443         
12444         var oldPre = this.inPre;
12445         if (node.tagName == 'PRE') {
12446             this.inPre = true;
12447         }
12448         for( var i = 0; i < node.childNodes.length; i++) {
12449             method.call(this, node.childNodes[i]);
12450         }
12451         this.inPre = oldPre;
12452     },
12453     
12454     
12455     
12456     /**
12457      * compile the template
12458      *
12459      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12460      *
12461      */
12462     compile: function()
12463     {
12464         var s = this.html;
12465         
12466         // covert the html into DOM...
12467         var doc = false;
12468         var div =false;
12469         try {
12470             doc = document.implementation.createHTMLDocument("");
12471             doc.documentElement.innerHTML =   this.html  ;
12472             div = doc.documentElement;
12473         } catch (e) {
12474             // old IE... - nasty -- it causes all sorts of issues.. with
12475             // images getting pulled from server..
12476             div = document.createElement('div');
12477             div.innerHTML = this.html;
12478         }
12479         //doc.documentElement.innerHTML = htmlBody
12480          
12481         
12482         
12483         this.tpls = [];
12484         var _t = this;
12485         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12486         
12487         var tpls = this.tpls;
12488         
12489         // create a top level template from the snippet..
12490         
12491         //Roo.log(div.innerHTML);
12492         
12493         var tpl = {
12494             uid : 'master',
12495             id : this.id++,
12496             attr : false,
12497             value : false,
12498             body : div.innerHTML,
12499             
12500             forCall : false,
12501             execCall : false,
12502             dom : div,
12503             isTop : true
12504             
12505         };
12506         tpls.unshift(tpl);
12507         
12508         
12509         // compile them...
12510         this.tpls = [];
12511         Roo.each(tpls, function(tp){
12512             this.compileTpl(tp);
12513             this.tpls[tp.id] = tp;
12514         }, this);
12515         
12516         this.master = tpls[0];
12517         return this;
12518         
12519         
12520     },
12521     
12522     compileNode : function(node, istop) {
12523         // test for
12524         //Roo.log(node);
12525         
12526         
12527         // skip anything not a tag..
12528         if (node.nodeType != 1) {
12529             if (node.nodeType == 3 && !this.inPre) {
12530                 // reduce white space..
12531                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12532                 
12533             }
12534             return;
12535         }
12536         
12537         var tpl = {
12538             uid : false,
12539             id : false,
12540             attr : false,
12541             value : false,
12542             body : '',
12543             
12544             forCall : false,
12545             execCall : false,
12546             dom : false,
12547             isTop : istop
12548             
12549             
12550         };
12551         
12552         
12553         switch(true) {
12554             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12555             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12556             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12557             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12558             // no default..
12559         }
12560         
12561         
12562         if (!tpl.attr) {
12563             // just itterate children..
12564             this.iterChild(node,this.compileNode);
12565             return;
12566         }
12567         tpl.uid = this.id++;
12568         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12569         node.removeAttribute('roo-'+ tpl.attr);
12570         if (tpl.attr != 'name') {
12571             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12572             node.parentNode.replaceChild(placeholder,  node);
12573         } else {
12574             
12575             var placeholder =  document.createElement('span');
12576             placeholder.className = 'roo-tpl-' + tpl.value;
12577             node.parentNode.replaceChild(placeholder,  node);
12578         }
12579         
12580         // parent now sees '{domtplXXXX}
12581         this.iterChild(node,this.compileNode);
12582         
12583         // we should now have node body...
12584         var div = document.createElement('div');
12585         div.appendChild(node);
12586         tpl.dom = node;
12587         // this has the unfortunate side effect of converting tagged attributes
12588         // eg. href="{...}" into %7C...%7D
12589         // this has been fixed by searching for those combo's although it's a bit hacky..
12590         
12591         
12592         tpl.body = div.innerHTML;
12593         
12594         
12595          
12596         tpl.id = tpl.uid;
12597         switch(tpl.attr) {
12598             case 'for' :
12599                 switch (tpl.value) {
12600                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12601                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12602                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12603                 }
12604                 break;
12605             
12606             case 'exec':
12607                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12608                 break;
12609             
12610             case 'if':     
12611                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12612                 break;
12613             
12614             case 'name':
12615                 tpl.id  = tpl.value; // replace non characters???
12616                 break;
12617             
12618         }
12619         
12620         
12621         this.tpls.push(tpl);
12622         
12623         
12624         
12625     },
12626     
12627     
12628     
12629     
12630     /**
12631      * Compile a segment of the template into a 'sub-template'
12632      *
12633      * 
12634      * 
12635      *
12636      */
12637     compileTpl : function(tpl)
12638     {
12639         var fm = Roo.util.Format;
12640         var useF = this.disableFormats !== true;
12641         
12642         var sep = Roo.isGecko ? "+\n" : ",\n";
12643         
12644         var undef = function(str) {
12645             Roo.debug && Roo.log("Property not found :"  + str);
12646             return '';
12647         };
12648           
12649         //Roo.log(tpl.body);
12650         
12651         
12652         
12653         var fn = function(m, lbrace, name, format, args)
12654         {
12655             //Roo.log("ARGS");
12656             //Roo.log(arguments);
12657             args = args ? args.replace(/\\'/g,"'") : args;
12658             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12659             if (typeof(format) == 'undefined') {
12660                 format =  'htmlEncode'; 
12661             }
12662             if (format == 'raw' ) {
12663                 format = false;
12664             }
12665             
12666             if(name.substr(0, 6) == 'domtpl'){
12667                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12668             }
12669             
12670             // build an array of options to determine if value is undefined..
12671             
12672             // basically get 'xxxx.yyyy' then do
12673             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12674             //    (function () { Roo.log("Property not found"); return ''; })() :
12675             //    ......
12676             
12677             var udef_ar = [];
12678             var lookfor = '';
12679             Roo.each(name.split('.'), function(st) {
12680                 lookfor += (lookfor.length ? '.': '') + st;
12681                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12682             });
12683             
12684             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12685             
12686             
12687             if(format && useF){
12688                 
12689                 args = args ? ',' + args : "";
12690                  
12691                 if(format.substr(0, 5) != "this."){
12692                     format = "fm." + format + '(';
12693                 }else{
12694                     format = 'this.call("'+ format.substr(5) + '", ';
12695                     args = ", values";
12696                 }
12697                 
12698                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12699             }
12700              
12701             if (args && args.length) {
12702                 // called with xxyx.yuu:(test,test)
12703                 // change to ()
12704                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12705             }
12706             // raw.. - :raw modifier..
12707             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12708             
12709         };
12710         var body;
12711         // branched to use + in gecko and [].join() in others
12712         if(Roo.isGecko){
12713             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12714                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12715                     "';};};";
12716         }else{
12717             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12718             body.push(tpl.body.replace(/(\r\n|\n)/g,
12719                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12720             body.push("'].join('');};};");
12721             body = body.join('');
12722         }
12723         
12724         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12725        
12726         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12727         eval(body);
12728         
12729         return this;
12730     },
12731      
12732     /**
12733      * same as applyTemplate, except it's done to one of the subTemplates
12734      * when using named templates, you can do:
12735      *
12736      * var str = pl.applySubTemplate('your-name', values);
12737      *
12738      * 
12739      * @param {Number} id of the template
12740      * @param {Object} values to apply to template
12741      * @param {Object} parent (normaly the instance of this object)
12742      */
12743     applySubTemplate : function(id, values, parent)
12744     {
12745         
12746         
12747         var t = this.tpls[id];
12748         
12749         
12750         try { 
12751             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12752                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12753                 return '';
12754             }
12755         } catch(e) {
12756             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12757             Roo.log(values);
12758           
12759             return '';
12760         }
12761         try { 
12762             
12763             if(t.execCall && t.execCall.call(this, values, parent)){
12764                 return '';
12765             }
12766         } catch(e) {
12767             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12768             Roo.log(values);
12769             return '';
12770         }
12771         
12772         try {
12773             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12774             parent = t.target ? values : parent;
12775             if(t.forCall && vs instanceof Array){
12776                 var buf = [];
12777                 for(var i = 0, len = vs.length; i < len; i++){
12778                     try {
12779                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12780                     } catch (e) {
12781                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12782                         Roo.log(e.body);
12783                         //Roo.log(t.compiled);
12784                         Roo.log(vs[i]);
12785                     }   
12786                 }
12787                 return buf.join('');
12788             }
12789         } catch (e) {
12790             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12791             Roo.log(values);
12792             return '';
12793         }
12794         try {
12795             return t.compiled.call(this, vs, parent);
12796         } catch (e) {
12797             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12798             Roo.log(e.body);
12799             //Roo.log(t.compiled);
12800             Roo.log(values);
12801             return '';
12802         }
12803     },
12804
12805    
12806
12807     applyTemplate : function(values){
12808         return this.master.compiled.call(this, values, {});
12809         //var s = this.subs;
12810     },
12811
12812     apply : function(){
12813         return this.applyTemplate.apply(this, arguments);
12814     }
12815
12816  });
12817
12818 Roo.DomTemplate.from = function(el){
12819     el = Roo.getDom(el);
12820     return new Roo.Domtemplate(el.value || el.innerHTML);
12821 };/*
12822  * Based on:
12823  * Ext JS Library 1.1.1
12824  * Copyright(c) 2006-2007, Ext JS, LLC.
12825  *
12826  * Originally Released Under LGPL - original licence link has changed is not relivant.
12827  *
12828  * Fork - LGPL
12829  * <script type="text/javascript">
12830  */
12831
12832 /**
12833  * @class Roo.util.DelayedTask
12834  * Provides a convenient method of performing setTimeout where a new
12835  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12836  * You can use this class to buffer
12837  * the keypress events for a certain number of milliseconds, and perform only if they stop
12838  * for that amount of time.
12839  * @constructor The parameters to this constructor serve as defaults and are not required.
12840  * @param {Function} fn (optional) The default function to timeout
12841  * @param {Object} scope (optional) The default scope of that timeout
12842  * @param {Array} args (optional) The default Array of arguments
12843  */
12844 Roo.util.DelayedTask = function(fn, scope, args){
12845     var id = null, d, t;
12846
12847     var call = function(){
12848         var now = new Date().getTime();
12849         if(now - t >= d){
12850             clearInterval(id);
12851             id = null;
12852             fn.apply(scope, args || []);
12853         }
12854     };
12855     /**
12856      * Cancels any pending timeout and queues a new one
12857      * @param {Number} delay The milliseconds to delay
12858      * @param {Function} newFn (optional) Overrides function passed to constructor
12859      * @param {Object} newScope (optional) Overrides scope passed to constructor
12860      * @param {Array} newArgs (optional) Overrides args passed to constructor
12861      */
12862     this.delay = function(delay, newFn, newScope, newArgs){
12863         if(id && delay != d){
12864             this.cancel();
12865         }
12866         d = delay;
12867         t = new Date().getTime();
12868         fn = newFn || fn;
12869         scope = newScope || scope;
12870         args = newArgs || args;
12871         if(!id){
12872             id = setInterval(call, d);
12873         }
12874     };
12875
12876     /**
12877      * Cancel the last queued timeout
12878      */
12879     this.cancel = function(){
12880         if(id){
12881             clearInterval(id);
12882             id = null;
12883         }
12884     };
12885 };/*
12886  * Based on:
12887  * Ext JS Library 1.1.1
12888  * Copyright(c) 2006-2007, Ext JS, LLC.
12889  *
12890  * Originally Released Under LGPL - original licence link has changed is not relivant.
12891  *
12892  * Fork - LGPL
12893  * <script type="text/javascript">
12894  */
12895  
12896  
12897 Roo.util.TaskRunner = function(interval){
12898     interval = interval || 10;
12899     var tasks = [], removeQueue = [];
12900     var id = 0;
12901     var running = false;
12902
12903     var stopThread = function(){
12904         running = false;
12905         clearInterval(id);
12906         id = 0;
12907     };
12908
12909     var startThread = function(){
12910         if(!running){
12911             running = true;
12912             id = setInterval(runTasks, interval);
12913         }
12914     };
12915
12916     var removeTask = function(task){
12917         removeQueue.push(task);
12918         if(task.onStop){
12919             task.onStop();
12920         }
12921     };
12922
12923     var runTasks = function(){
12924         if(removeQueue.length > 0){
12925             for(var i = 0, len = removeQueue.length; i < len; i++){
12926                 tasks.remove(removeQueue[i]);
12927             }
12928             removeQueue = [];
12929             if(tasks.length < 1){
12930                 stopThread();
12931                 return;
12932             }
12933         }
12934         var now = new Date().getTime();
12935         for(var i = 0, len = tasks.length; i < len; ++i){
12936             var t = tasks[i];
12937             var itime = now - t.taskRunTime;
12938             if(t.interval <= itime){
12939                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12940                 t.taskRunTime = now;
12941                 if(rt === false || t.taskRunCount === t.repeat){
12942                     removeTask(t);
12943                     return;
12944                 }
12945             }
12946             if(t.duration && t.duration <= (now - t.taskStartTime)){
12947                 removeTask(t);
12948             }
12949         }
12950     };
12951
12952     /**
12953      * Queues a new task.
12954      * @param {Object} task
12955      */
12956     this.start = function(task){
12957         tasks.push(task);
12958         task.taskStartTime = new Date().getTime();
12959         task.taskRunTime = 0;
12960         task.taskRunCount = 0;
12961         startThread();
12962         return task;
12963     };
12964
12965     this.stop = function(task){
12966         removeTask(task);
12967         return task;
12968     };
12969
12970     this.stopAll = function(){
12971         stopThread();
12972         for(var i = 0, len = tasks.length; i < len; i++){
12973             if(tasks[i].onStop){
12974                 tasks[i].onStop();
12975             }
12976         }
12977         tasks = [];
12978         removeQueue = [];
12979     };
12980 };
12981
12982 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12983  * Based on:
12984  * Ext JS Library 1.1.1
12985  * Copyright(c) 2006-2007, Ext JS, LLC.
12986  *
12987  * Originally Released Under LGPL - original licence link has changed is not relivant.
12988  *
12989  * Fork - LGPL
12990  * <script type="text/javascript">
12991  */
12992
12993  
12994 /**
12995  * @class Roo.util.MixedCollection
12996  * @extends Roo.util.Observable
12997  * A Collection class that maintains both numeric indexes and keys and exposes events.
12998  * @constructor
12999  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
13000  * collection (defaults to false)
13001  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
13002  * and return the key value for that item.  This is used when available to look up the key on items that
13003  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
13004  * equivalent to providing an implementation for the {@link #getKey} method.
13005  */
13006 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13007     this.items = [];
13008     this.map = {};
13009     this.keys = [];
13010     this.length = 0;
13011     this.addEvents({
13012         /**
13013          * @event clear
13014          * Fires when the collection is cleared.
13015          */
13016         "clear" : true,
13017         /**
13018          * @event add
13019          * Fires when an item is added to the collection.
13020          * @param {Number} index The index at which the item was added.
13021          * @param {Object} o The item added.
13022          * @param {String} key The key associated with the added item.
13023          */
13024         "add" : true,
13025         /**
13026          * @event replace
13027          * Fires when an item is replaced in the collection.
13028          * @param {String} key he key associated with the new added.
13029          * @param {Object} old The item being replaced.
13030          * @param {Object} new The new item.
13031          */
13032         "replace" : true,
13033         /**
13034          * @event remove
13035          * Fires when an item is removed from the collection.
13036          * @param {Object} o The item being removed.
13037          * @param {String} key (optional) The key associated with the removed item.
13038          */
13039         "remove" : true,
13040         "sort" : true
13041     });
13042     this.allowFunctions = allowFunctions === true;
13043     if(keyFn){
13044         this.getKey = keyFn;
13045     }
13046     Roo.util.MixedCollection.superclass.constructor.call(this);
13047 };
13048
13049 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13050     allowFunctions : false,
13051     
13052 /**
13053  * Adds an item to the collection.
13054  * @param {String} key The key to associate with the item
13055  * @param {Object} o The item to add.
13056  * @return {Object} The item added.
13057  */
13058     add : function(key, o){
13059         if(arguments.length == 1){
13060             o = arguments[0];
13061             key = this.getKey(o);
13062         }
13063         if(typeof key == "undefined" || key === null){
13064             this.length++;
13065             this.items.push(o);
13066             this.keys.push(null);
13067         }else{
13068             var old = this.map[key];
13069             if(old){
13070                 return this.replace(key, o);
13071             }
13072             this.length++;
13073             this.items.push(o);
13074             this.map[key] = o;
13075             this.keys.push(key);
13076         }
13077         this.fireEvent("add", this.length-1, o, key);
13078         return o;
13079     },
13080        
13081 /**
13082   * MixedCollection has a generic way to fetch keys if you implement getKey.
13083 <pre><code>
13084 // normal way
13085 var mc = new Roo.util.MixedCollection();
13086 mc.add(someEl.dom.id, someEl);
13087 mc.add(otherEl.dom.id, otherEl);
13088 //and so on
13089
13090 // using getKey
13091 var mc = new Roo.util.MixedCollection();
13092 mc.getKey = function(el){
13093    return el.dom.id;
13094 };
13095 mc.add(someEl);
13096 mc.add(otherEl);
13097
13098 // or via the constructor
13099 var mc = new Roo.util.MixedCollection(false, function(el){
13100    return el.dom.id;
13101 });
13102 mc.add(someEl);
13103 mc.add(otherEl);
13104 </code></pre>
13105  * @param o {Object} The item for which to find the key.
13106  * @return {Object} The key for the passed item.
13107  */
13108     getKey : function(o){
13109          return o.id; 
13110     },
13111    
13112 /**
13113  * Replaces an item in the collection.
13114  * @param {String} key The key associated with the item to replace, or the item to replace.
13115  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13116  * @return {Object}  The new item.
13117  */
13118     replace : function(key, o){
13119         if(arguments.length == 1){
13120             o = arguments[0];
13121             key = this.getKey(o);
13122         }
13123         var old = this.item(key);
13124         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13125              return this.add(key, o);
13126         }
13127         var index = this.indexOfKey(key);
13128         this.items[index] = o;
13129         this.map[key] = o;
13130         this.fireEvent("replace", key, old, o);
13131         return o;
13132     },
13133    
13134 /**
13135  * Adds all elements of an Array or an Object to the collection.
13136  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13137  * an Array of values, each of which are added to the collection.
13138  */
13139     addAll : function(objs){
13140         if(arguments.length > 1 || objs instanceof Array){
13141             var args = arguments.length > 1 ? arguments : objs;
13142             for(var i = 0, len = args.length; i < len; i++){
13143                 this.add(args[i]);
13144             }
13145         }else{
13146             for(var key in objs){
13147                 if(this.allowFunctions || typeof objs[key] != "function"){
13148                     this.add(key, objs[key]);
13149                 }
13150             }
13151         }
13152     },
13153    
13154 /**
13155  * Executes the specified function once for every item in the collection, passing each
13156  * item as the first and only parameter. returning false from the function will stop the iteration.
13157  * @param {Function} fn The function to execute for each item.
13158  * @param {Object} scope (optional) The scope in which to execute the function.
13159  */
13160     each : function(fn, scope){
13161         var items = [].concat(this.items); // each safe for removal
13162         for(var i = 0, len = items.length; i < len; i++){
13163             if(fn.call(scope || items[i], items[i], i, len) === false){
13164                 break;
13165             }
13166         }
13167     },
13168    
13169 /**
13170  * Executes the specified function once for every key in the collection, passing each
13171  * key, and its associated item as the first two parameters.
13172  * @param {Function} fn The function to execute for each item.
13173  * @param {Object} scope (optional) The scope in which to execute the function.
13174  */
13175     eachKey : function(fn, scope){
13176         for(var i = 0, len = this.keys.length; i < len; i++){
13177             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13178         }
13179     },
13180    
13181 /**
13182  * Returns the first item in the collection which elicits a true return value from the
13183  * passed selection function.
13184  * @param {Function} fn The selection function to execute for each item.
13185  * @param {Object} scope (optional) The scope in which to execute the function.
13186  * @return {Object} The first item in the collection which returned true from the selection function.
13187  */
13188     find : function(fn, scope){
13189         for(var i = 0, len = this.items.length; i < len; i++){
13190             if(fn.call(scope || window, this.items[i], this.keys[i])){
13191                 return this.items[i];
13192             }
13193         }
13194         return null;
13195     },
13196    
13197 /**
13198  * Inserts an item at the specified index in the collection.
13199  * @param {Number} index The index to insert the item at.
13200  * @param {String} key The key to associate with the new item, or the item itself.
13201  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13202  * @return {Object} The item inserted.
13203  */
13204     insert : function(index, key, o){
13205         if(arguments.length == 2){
13206             o = arguments[1];
13207             key = this.getKey(o);
13208         }
13209         if(index >= this.length){
13210             return this.add(key, o);
13211         }
13212         this.length++;
13213         this.items.splice(index, 0, o);
13214         if(typeof key != "undefined" && key != null){
13215             this.map[key] = o;
13216         }
13217         this.keys.splice(index, 0, key);
13218         this.fireEvent("add", index, o, key);
13219         return o;
13220     },
13221    
13222 /**
13223  * Removed an item from the collection.
13224  * @param {Object} o The item to remove.
13225  * @return {Object} The item removed.
13226  */
13227     remove : function(o){
13228         return this.removeAt(this.indexOf(o));
13229     },
13230    
13231 /**
13232  * Remove an item from a specified index in the collection.
13233  * @param {Number} index The index within the collection of the item to remove.
13234  */
13235     removeAt : function(index){
13236         if(index < this.length && index >= 0){
13237             this.length--;
13238             var o = this.items[index];
13239             this.items.splice(index, 1);
13240             var key = this.keys[index];
13241             if(typeof key != "undefined"){
13242                 delete this.map[key];
13243             }
13244             this.keys.splice(index, 1);
13245             this.fireEvent("remove", o, key);
13246         }
13247     },
13248    
13249 /**
13250  * Removed an item associated with the passed key fom the collection.
13251  * @param {String} key The key of the item to remove.
13252  */
13253     removeKey : function(key){
13254         return this.removeAt(this.indexOfKey(key));
13255     },
13256    
13257 /**
13258  * Returns the number of items in the collection.
13259  * @return {Number} the number of items in the collection.
13260  */
13261     getCount : function(){
13262         return this.length; 
13263     },
13264    
13265 /**
13266  * Returns index within the collection of the passed Object.
13267  * @param {Object} o The item to find the index of.
13268  * @return {Number} index of the item.
13269  */
13270     indexOf : function(o){
13271         if(!this.items.indexOf){
13272             for(var i = 0, len = this.items.length; i < len; i++){
13273                 if(this.items[i] == o) {
13274                     return i;
13275                 }
13276             }
13277             return -1;
13278         }else{
13279             return this.items.indexOf(o);
13280         }
13281     },
13282    
13283 /**
13284  * Returns index within the collection of the passed key.
13285  * @param {String} key The key to find the index of.
13286  * @return {Number} index of the key.
13287  */
13288     indexOfKey : function(key){
13289         if(!this.keys.indexOf){
13290             for(var i = 0, len = this.keys.length; i < len; i++){
13291                 if(this.keys[i] == key) {
13292                     return i;
13293                 }
13294             }
13295             return -1;
13296         }else{
13297             return this.keys.indexOf(key);
13298         }
13299     },
13300    
13301 /**
13302  * Returns the item associated with the passed key OR index. Key has priority over index.
13303  * @param {String/Number} key The key or index of the item.
13304  * @return {Object} The item associated with the passed key.
13305  */
13306     item : function(key){
13307         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13308         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13309     },
13310     
13311 /**
13312  * Returns the item at the specified index.
13313  * @param {Number} index The index of the item.
13314  * @return {Object}
13315  */
13316     itemAt : function(index){
13317         return this.items[index];
13318     },
13319     
13320 /**
13321  * Returns the item associated with the passed key.
13322  * @param {String/Number} key The key of the item.
13323  * @return {Object} The item associated with the passed key.
13324  */
13325     key : function(key){
13326         return this.map[key];
13327     },
13328    
13329 /**
13330  * Returns true if the collection contains the passed Object as an item.
13331  * @param {Object} o  The Object to look for in the collection.
13332  * @return {Boolean} True if the collection contains the Object as an item.
13333  */
13334     contains : function(o){
13335         return this.indexOf(o) != -1;
13336     },
13337    
13338 /**
13339  * Returns true if the collection contains the passed Object as a key.
13340  * @param {String} key The key to look for in the collection.
13341  * @return {Boolean} True if the collection contains the Object as a key.
13342  */
13343     containsKey : function(key){
13344         return typeof this.map[key] != "undefined";
13345     },
13346    
13347 /**
13348  * Removes all items from the collection.
13349  */
13350     clear : function(){
13351         this.length = 0;
13352         this.items = [];
13353         this.keys = [];
13354         this.map = {};
13355         this.fireEvent("clear");
13356     },
13357    
13358 /**
13359  * Returns the first item in the collection.
13360  * @return {Object} the first item in the collection..
13361  */
13362     first : function(){
13363         return this.items[0]; 
13364     },
13365    
13366 /**
13367  * Returns the last item in the collection.
13368  * @return {Object} the last item in the collection..
13369  */
13370     last : function(){
13371         return this.items[this.length-1];   
13372     },
13373     
13374     _sort : function(property, dir, fn){
13375         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13376         fn = fn || function(a, b){
13377             return a-b;
13378         };
13379         var c = [], k = this.keys, items = this.items;
13380         for(var i = 0, len = items.length; i < len; i++){
13381             c[c.length] = {key: k[i], value: items[i], index: i};
13382         }
13383         c.sort(function(a, b){
13384             var v = fn(a[property], b[property]) * dsc;
13385             if(v == 0){
13386                 v = (a.index < b.index ? -1 : 1);
13387             }
13388             return v;
13389         });
13390         for(var i = 0, len = c.length; i < len; i++){
13391             items[i] = c[i].value;
13392             k[i] = c[i].key;
13393         }
13394         this.fireEvent("sort", this);
13395     },
13396     
13397     /**
13398      * Sorts this collection with the passed comparison function
13399      * @param {String} direction (optional) "ASC" or "DESC"
13400      * @param {Function} fn (optional) comparison function
13401      */
13402     sort : function(dir, fn){
13403         this._sort("value", dir, fn);
13404     },
13405     
13406     /**
13407      * Sorts this collection by keys
13408      * @param {String} direction (optional) "ASC" or "DESC"
13409      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13410      */
13411     keySort : function(dir, fn){
13412         this._sort("key", dir, fn || function(a, b){
13413             return String(a).toUpperCase()-String(b).toUpperCase();
13414         });
13415     },
13416     
13417     /**
13418      * Returns a range of items in this collection
13419      * @param {Number} startIndex (optional) defaults to 0
13420      * @param {Number} endIndex (optional) default to the last item
13421      * @return {Array} An array of items
13422      */
13423     getRange : function(start, end){
13424         var items = this.items;
13425         if(items.length < 1){
13426             return [];
13427         }
13428         start = start || 0;
13429         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13430         var r = [];
13431         if(start <= end){
13432             for(var i = start; i <= end; i++) {
13433                     r[r.length] = items[i];
13434             }
13435         }else{
13436             for(var i = start; i >= end; i--) {
13437                     r[r.length] = items[i];
13438             }
13439         }
13440         return r;
13441     },
13442         
13443     /**
13444      * Filter the <i>objects</i> in this collection by a specific property. 
13445      * Returns a new collection that has been filtered.
13446      * @param {String} property A property on your objects
13447      * @param {String/RegExp} value Either string that the property values 
13448      * should start with or a RegExp to test against the property
13449      * @return {MixedCollection} The new filtered collection
13450      */
13451     filter : function(property, value){
13452         if(!value.exec){ // not a regex
13453             value = String(value);
13454             if(value.length == 0){
13455                 return this.clone();
13456             }
13457             value = new RegExp("^" + Roo.escapeRe(value), "i");
13458         }
13459         return this.filterBy(function(o){
13460             return o && value.test(o[property]);
13461         });
13462         },
13463     
13464     /**
13465      * Filter by a function. * Returns a new collection that has been filtered.
13466      * The passed function will be called with each 
13467      * object in the collection. If the function returns true, the value is included 
13468      * otherwise it is filtered.
13469      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13470      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13471      * @return {MixedCollection} The new filtered collection
13472      */
13473     filterBy : function(fn, scope){
13474         var r = new Roo.util.MixedCollection();
13475         r.getKey = this.getKey;
13476         var k = this.keys, it = this.items;
13477         for(var i = 0, len = it.length; i < len; i++){
13478             if(fn.call(scope||this, it[i], k[i])){
13479                                 r.add(k[i], it[i]);
13480                         }
13481         }
13482         return r;
13483     },
13484     
13485     /**
13486      * Creates a duplicate of this collection
13487      * @return {MixedCollection}
13488      */
13489     clone : function(){
13490         var r = new Roo.util.MixedCollection();
13491         var k = this.keys, it = this.items;
13492         for(var i = 0, len = it.length; i < len; i++){
13493             r.add(k[i], it[i]);
13494         }
13495         r.getKey = this.getKey;
13496         return r;
13497     }
13498 });
13499 /**
13500  * Returns the item associated with the passed key or index.
13501  * @method
13502  * @param {String/Number} key The key or index of the item.
13503  * @return {Object} The item associated with the passed key.
13504  */
13505 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13506  * Based on:
13507  * Ext JS Library 1.1.1
13508  * Copyright(c) 2006-2007, Ext JS, LLC.
13509  *
13510  * Originally Released Under LGPL - original licence link has changed is not relivant.
13511  *
13512  * Fork - LGPL
13513  * <script type="text/javascript">
13514  */
13515 /**
13516  * @class Roo.util.JSON
13517  * Modified version of Douglas Crockford"s json.js that doesn"t
13518  * mess with the Object prototype 
13519  * http://www.json.org/js.html
13520  * @singleton
13521  */
13522 Roo.util.JSON = new (function(){
13523     var useHasOwn = {}.hasOwnProperty ? true : false;
13524     
13525     // crashes Safari in some instances
13526     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13527     
13528     var pad = function(n) {
13529         return n < 10 ? "0" + n : n;
13530     };
13531     
13532     var m = {
13533         "\b": '\\b',
13534         "\t": '\\t',
13535         "\n": '\\n',
13536         "\f": '\\f',
13537         "\r": '\\r',
13538         '"' : '\\"',
13539         "\\": '\\\\'
13540     };
13541
13542     var encodeString = function(s){
13543         if (/["\\\x00-\x1f]/.test(s)) {
13544             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13545                 var c = m[b];
13546                 if(c){
13547                     return c;
13548                 }
13549                 c = b.charCodeAt();
13550                 return "\\u00" +
13551                     Math.floor(c / 16).toString(16) +
13552                     (c % 16).toString(16);
13553             }) + '"';
13554         }
13555         return '"' + s + '"';
13556     };
13557     
13558     var encodeArray = function(o){
13559         var a = ["["], b, i, l = o.length, v;
13560             for (i = 0; i < l; i += 1) {
13561                 v = o[i];
13562                 switch (typeof v) {
13563                     case "undefined":
13564                     case "function":
13565                     case "unknown":
13566                         break;
13567                     default:
13568                         if (b) {
13569                             a.push(',');
13570                         }
13571                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13572                         b = true;
13573                 }
13574             }
13575             a.push("]");
13576             return a.join("");
13577     };
13578     
13579     var encodeDate = function(o){
13580         return '"' + o.getFullYear() + "-" +
13581                 pad(o.getMonth() + 1) + "-" +
13582                 pad(o.getDate()) + "T" +
13583                 pad(o.getHours()) + ":" +
13584                 pad(o.getMinutes()) + ":" +
13585                 pad(o.getSeconds()) + '"';
13586     };
13587     
13588     /**
13589      * Encodes an Object, Array or other value
13590      * @param {Mixed} o The variable to encode
13591      * @return {String} The JSON string
13592      */
13593     this.encode = function(o)
13594     {
13595         // should this be extended to fully wrap stringify..
13596         
13597         if(typeof o == "undefined" || o === null){
13598             return "null";
13599         }else if(o instanceof Array){
13600             return encodeArray(o);
13601         }else if(o instanceof Date){
13602             return encodeDate(o);
13603         }else if(typeof o == "string"){
13604             return encodeString(o);
13605         }else if(typeof o == "number"){
13606             return isFinite(o) ? String(o) : "null";
13607         }else if(typeof o == "boolean"){
13608             return String(o);
13609         }else {
13610             var a = ["{"], b, i, v;
13611             for (i in o) {
13612                 if(!useHasOwn || o.hasOwnProperty(i)) {
13613                     v = o[i];
13614                     switch (typeof v) {
13615                     case "undefined":
13616                     case "function":
13617                     case "unknown":
13618                         break;
13619                     default:
13620                         if(b){
13621                             a.push(',');
13622                         }
13623                         a.push(this.encode(i), ":",
13624                                 v === null ? "null" : this.encode(v));
13625                         b = true;
13626                     }
13627                 }
13628             }
13629             a.push("}");
13630             return a.join("");
13631         }
13632     };
13633     
13634     /**
13635      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13636      * @param {String} json The JSON string
13637      * @return {Object} The resulting object
13638      */
13639     this.decode = function(json){
13640         
13641         return  /** eval:var:json */ eval("(" + json + ')');
13642     };
13643 })();
13644 /** 
13645  * Shorthand for {@link Roo.util.JSON#encode}
13646  * @member Roo encode 
13647  * @method */
13648 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13649 /** 
13650  * Shorthand for {@link Roo.util.JSON#decode}
13651  * @member Roo decode 
13652  * @method */
13653 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13654 /*
13655  * Based on:
13656  * Ext JS Library 1.1.1
13657  * Copyright(c) 2006-2007, Ext JS, LLC.
13658  *
13659  * Originally Released Under LGPL - original licence link has changed is not relivant.
13660  *
13661  * Fork - LGPL
13662  * <script type="text/javascript">
13663  */
13664  
13665 /**
13666  * @class Roo.util.Format
13667  * Reusable data formatting functions
13668  * @singleton
13669  */
13670 Roo.util.Format = function(){
13671     var trimRe = /^\s+|\s+$/g;
13672     return {
13673         /**
13674          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13675          * @param {String} value The string to truncate
13676          * @param {Number} length The maximum length to allow before truncating
13677          * @return {String} The converted text
13678          */
13679         ellipsis : function(value, len){
13680             if(value && value.length > len){
13681                 return value.substr(0, len-3)+"...";
13682             }
13683             return value;
13684         },
13685
13686         /**
13687          * Checks a reference and converts it to empty string if it is undefined
13688          * @param {Mixed} value Reference to check
13689          * @return {Mixed} Empty string if converted, otherwise the original value
13690          */
13691         undef : function(value){
13692             return typeof value != "undefined" ? value : "";
13693         },
13694
13695         /**
13696          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13697          * @param {String} value The string to encode
13698          * @return {String} The encoded text
13699          */
13700         htmlEncode : function(value){
13701             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13702         },
13703
13704         /**
13705          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13706          * @param {String} value The string to decode
13707          * @return {String} The decoded text
13708          */
13709         htmlDecode : function(value){
13710             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13711         },
13712
13713         /**
13714          * Trims any whitespace from either side of a string
13715          * @param {String} value The text to trim
13716          * @return {String} The trimmed text
13717          */
13718         trim : function(value){
13719             return String(value).replace(trimRe, "");
13720         },
13721
13722         /**
13723          * Returns a substring from within an original string
13724          * @param {String} value The original text
13725          * @param {Number} start The start index of the substring
13726          * @param {Number} length The length of the substring
13727          * @return {String} The substring
13728          */
13729         substr : function(value, start, length){
13730             return String(value).substr(start, length);
13731         },
13732
13733         /**
13734          * Converts a string to all lower case letters
13735          * @param {String} value The text to convert
13736          * @return {String} The converted text
13737          */
13738         lowercase : function(value){
13739             return String(value).toLowerCase();
13740         },
13741
13742         /**
13743          * Converts a string to all upper case letters
13744          * @param {String} value The text to convert
13745          * @return {String} The converted text
13746          */
13747         uppercase : function(value){
13748             return String(value).toUpperCase();
13749         },
13750
13751         /**
13752          * Converts the first character only of a string to upper case
13753          * @param {String} value The text to convert
13754          * @return {String} The converted text
13755          */
13756         capitalize : function(value){
13757             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13758         },
13759
13760         // private
13761         call : function(value, fn){
13762             if(arguments.length > 2){
13763                 var args = Array.prototype.slice.call(arguments, 2);
13764                 args.unshift(value);
13765                  
13766                 return /** eval:var:value */  eval(fn).apply(window, args);
13767             }else{
13768                 /** eval:var:value */
13769                 return /** eval:var:value */ eval(fn).call(window, value);
13770             }
13771         },
13772
13773        
13774         /**
13775          * safer version of Math.toFixed..??/
13776          * @param {Number/String} value The numeric value to format
13777          * @param {Number/String} value Decimal places 
13778          * @return {String} The formatted currency string
13779          */
13780         toFixed : function(v, n)
13781         {
13782             // why not use to fixed - precision is buggered???
13783             if (!n) {
13784                 return Math.round(v-0);
13785             }
13786             var fact = Math.pow(10,n+1);
13787             v = (Math.round((v-0)*fact))/fact;
13788             var z = (''+fact).substring(2);
13789             if (v == Math.floor(v)) {
13790                 return Math.floor(v) + '.' + z;
13791             }
13792             
13793             // now just padd decimals..
13794             var ps = String(v).split('.');
13795             var fd = (ps[1] + z);
13796             var r = fd.substring(0,n); 
13797             var rm = fd.substring(n); 
13798             if (rm < 5) {
13799                 return ps[0] + '.' + r;
13800             }
13801             r*=1; // turn it into a number;
13802             r++;
13803             if (String(r).length != n) {
13804                 ps[0]*=1;
13805                 ps[0]++;
13806                 r = String(r).substring(1); // chop the end off.
13807             }
13808             
13809             return ps[0] + '.' + r;
13810              
13811         },
13812         
13813         /**
13814          * Format a number as US currency
13815          * @param {Number/String} value The numeric value to format
13816          * @return {String} The formatted currency string
13817          */
13818         usMoney : function(v){
13819             return '$' + Roo.util.Format.number(v);
13820         },
13821         
13822         /**
13823          * Format a number
13824          * eventually this should probably emulate php's number_format
13825          * @param {Number/String} value The numeric value to format
13826          * @param {Number} decimals number of decimal places
13827          * @param {String} delimiter for thousands (default comma)
13828          * @return {String} The formatted currency string
13829          */
13830         number : function(v, decimals, thousandsDelimiter)
13831         {
13832             // multiply and round.
13833             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13834             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13835             
13836             var mul = Math.pow(10, decimals);
13837             var zero = String(mul).substring(1);
13838             v = (Math.round((v-0)*mul))/mul;
13839             
13840             // if it's '0' number.. then
13841             
13842             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13843             v = String(v);
13844             var ps = v.split('.');
13845             var whole = ps[0];
13846             
13847             var r = /(\d+)(\d{3})/;
13848             // add comma's
13849             
13850             if(thousandsDelimiter.length != 0) {
13851                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13852             } 
13853             
13854             var sub = ps[1] ?
13855                     // has decimals..
13856                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13857                     // does not have decimals
13858                     (decimals ? ('.' + zero) : '');
13859             
13860             
13861             return whole + sub ;
13862         },
13863         
13864         /**
13865          * Parse a value into a formatted date using the specified format pattern.
13866          * @param {Mixed} value The value to format
13867          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13868          * @return {String} The formatted date string
13869          */
13870         date : function(v, format){
13871             if(!v){
13872                 return "";
13873             }
13874             if(!(v instanceof Date)){
13875                 v = new Date(Date.parse(v));
13876             }
13877             return v.dateFormat(format || Roo.util.Format.defaults.date);
13878         },
13879
13880         /**
13881          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13882          * @param {String} format Any valid date format string
13883          * @return {Function} The date formatting function
13884          */
13885         dateRenderer : function(format){
13886             return function(v){
13887                 return Roo.util.Format.date(v, format);  
13888             };
13889         },
13890
13891         // private
13892         stripTagsRE : /<\/?[^>]+>/gi,
13893         
13894         /**
13895          * Strips all HTML tags
13896          * @param {Mixed} value The text from which to strip tags
13897          * @return {String} The stripped text
13898          */
13899         stripTags : function(v){
13900             return !v ? v : String(v).replace(this.stripTagsRE, "");
13901         }
13902     };
13903 }();
13904 Roo.util.Format.defaults = {
13905     date : 'd/M/Y'
13906 };/*
13907  * Based on:
13908  * Ext JS Library 1.1.1
13909  * Copyright(c) 2006-2007, Ext JS, LLC.
13910  *
13911  * Originally Released Under LGPL - original licence link has changed is not relivant.
13912  *
13913  * Fork - LGPL
13914  * <script type="text/javascript">
13915  */
13916
13917
13918  
13919
13920 /**
13921  * @class Roo.MasterTemplate
13922  * @extends Roo.Template
13923  * Provides a template that can have child templates. The syntax is:
13924 <pre><code>
13925 var t = new Roo.MasterTemplate(
13926         '&lt;select name="{name}"&gt;',
13927                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13928         '&lt;/select&gt;'
13929 );
13930 t.add('options', {value: 'foo', text: 'bar'});
13931 // or you can add multiple child elements in one shot
13932 t.addAll('options', [
13933     {value: 'foo', text: 'bar'},
13934     {value: 'foo2', text: 'bar2'},
13935     {value: 'foo3', text: 'bar3'}
13936 ]);
13937 // then append, applying the master template values
13938 t.append('my-form', {name: 'my-select'});
13939 </code></pre>
13940 * A name attribute for the child template is not required if you have only one child
13941 * template or you want to refer to them by index.
13942  */
13943 Roo.MasterTemplate = function(){
13944     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13945     this.originalHtml = this.html;
13946     var st = {};
13947     var m, re = this.subTemplateRe;
13948     re.lastIndex = 0;
13949     var subIndex = 0;
13950     while(m = re.exec(this.html)){
13951         var name = m[1], content = m[2];
13952         st[subIndex] = {
13953             name: name,
13954             index: subIndex,
13955             buffer: [],
13956             tpl : new Roo.Template(content)
13957         };
13958         if(name){
13959             st[name] = st[subIndex];
13960         }
13961         st[subIndex].tpl.compile();
13962         st[subIndex].tpl.call = this.call.createDelegate(this);
13963         subIndex++;
13964     }
13965     this.subCount = subIndex;
13966     this.subs = st;
13967 };
13968 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13969     /**
13970     * The regular expression used to match sub templates
13971     * @type RegExp
13972     * @property
13973     */
13974     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13975
13976     /**
13977      * Applies the passed values to a child template.
13978      * @param {String/Number} name (optional) The name or index of the child template
13979      * @param {Array/Object} values The values to be applied to the template
13980      * @return {MasterTemplate} this
13981      */
13982      add : function(name, values){
13983         if(arguments.length == 1){
13984             values = arguments[0];
13985             name = 0;
13986         }
13987         var s = this.subs[name];
13988         s.buffer[s.buffer.length] = s.tpl.apply(values);
13989         return this;
13990     },
13991
13992     /**
13993      * Applies all the passed values to a child template.
13994      * @param {String/Number} name (optional) The name or index of the child template
13995      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13996      * @param {Boolean} reset (optional) True to reset the template first
13997      * @return {MasterTemplate} this
13998      */
13999     fill : function(name, values, reset){
14000         var a = arguments;
14001         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
14002             values = a[0];
14003             name = 0;
14004             reset = a[1];
14005         }
14006         if(reset){
14007             this.reset();
14008         }
14009         for(var i = 0, len = values.length; i < len; i++){
14010             this.add(name, values[i]);
14011         }
14012         return this;
14013     },
14014
14015     /**
14016      * Resets the template for reuse
14017      * @return {MasterTemplate} this
14018      */
14019      reset : function(){
14020         var s = this.subs;
14021         for(var i = 0; i < this.subCount; i++){
14022             s[i].buffer = [];
14023         }
14024         return this;
14025     },
14026
14027     applyTemplate : function(values){
14028         var s = this.subs;
14029         var replaceIndex = -1;
14030         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14031             return s[++replaceIndex].buffer.join("");
14032         });
14033         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14034     },
14035
14036     apply : function(){
14037         return this.applyTemplate.apply(this, arguments);
14038     },
14039
14040     compile : function(){return this;}
14041 });
14042
14043 /**
14044  * Alias for fill().
14045  * @method
14046  */
14047 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14048  /**
14049  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14050  * var tpl = Roo.MasterTemplate.from('element-id');
14051  * @param {String/HTMLElement} el
14052  * @param {Object} config
14053  * @static
14054  */
14055 Roo.MasterTemplate.from = function(el, config){
14056     el = Roo.getDom(el);
14057     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14058 };/*
14059  * Based on:
14060  * Ext JS Library 1.1.1
14061  * Copyright(c) 2006-2007, Ext JS, LLC.
14062  *
14063  * Originally Released Under LGPL - original licence link has changed is not relivant.
14064  *
14065  * Fork - LGPL
14066  * <script type="text/javascript">
14067  */
14068
14069  
14070 /**
14071  * @class Roo.util.CSS
14072  * Utility class for manipulating CSS rules
14073  * @singleton
14074  */
14075 Roo.util.CSS = function(){
14076         var rules = null;
14077         var doc = document;
14078
14079     var camelRe = /(-[a-z])/gi;
14080     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14081
14082    return {
14083    /**
14084     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14085     * tag and appended to the HEAD of the document.
14086     * @param {String|Object} cssText The text containing the css rules
14087     * @param {String} id An id to add to the stylesheet for later removal
14088     * @return {StyleSheet}
14089     */
14090     createStyleSheet : function(cssText, id){
14091         var ss;
14092         var head = doc.getElementsByTagName("head")[0];
14093         var nrules = doc.createElement("style");
14094         nrules.setAttribute("type", "text/css");
14095         if(id){
14096             nrules.setAttribute("id", id);
14097         }
14098         if (typeof(cssText) != 'string') {
14099             // support object maps..
14100             // not sure if this a good idea.. 
14101             // perhaps it should be merged with the general css handling
14102             // and handle js style props.
14103             var cssTextNew = [];
14104             for(var n in cssText) {
14105                 var citems = [];
14106                 for(var k in cssText[n]) {
14107                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14108                 }
14109                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14110                 
14111             }
14112             cssText = cssTextNew.join("\n");
14113             
14114         }
14115        
14116        
14117        if(Roo.isIE){
14118            head.appendChild(nrules);
14119            ss = nrules.styleSheet;
14120            ss.cssText = cssText;
14121        }else{
14122            try{
14123                 nrules.appendChild(doc.createTextNode(cssText));
14124            }catch(e){
14125                nrules.cssText = cssText; 
14126            }
14127            head.appendChild(nrules);
14128            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14129        }
14130        this.cacheStyleSheet(ss);
14131        return ss;
14132    },
14133
14134    /**
14135     * Removes a style or link tag by id
14136     * @param {String} id The id of the tag
14137     */
14138    removeStyleSheet : function(id){
14139        var existing = doc.getElementById(id);
14140        if(existing){
14141            existing.parentNode.removeChild(existing);
14142        }
14143    },
14144
14145    /**
14146     * Dynamically swaps an existing stylesheet reference for a new one
14147     * @param {String} id The id of an existing link tag to remove
14148     * @param {String} url The href of the new stylesheet to include
14149     */
14150    swapStyleSheet : function(id, url){
14151        this.removeStyleSheet(id);
14152        var ss = doc.createElement("link");
14153        ss.setAttribute("rel", "stylesheet");
14154        ss.setAttribute("type", "text/css");
14155        ss.setAttribute("id", id);
14156        ss.setAttribute("href", url);
14157        doc.getElementsByTagName("head")[0].appendChild(ss);
14158    },
14159    
14160    /**
14161     * Refresh the rule cache if you have dynamically added stylesheets
14162     * @return {Object} An object (hash) of rules indexed by selector
14163     */
14164    refreshCache : function(){
14165        return this.getRules(true);
14166    },
14167
14168    // private
14169    cacheStyleSheet : function(stylesheet){
14170        if(!rules){
14171            rules = {};
14172        }
14173        try{// try catch for cross domain access issue
14174            var ssRules = stylesheet.cssRules || stylesheet.rules;
14175            for(var j = ssRules.length-1; j >= 0; --j){
14176                rules[ssRules[j].selectorText] = ssRules[j];
14177            }
14178        }catch(e){}
14179    },
14180    
14181    /**
14182     * Gets all css rules for the document
14183     * @param {Boolean} refreshCache true to refresh the internal cache
14184     * @return {Object} An object (hash) of rules indexed by selector
14185     */
14186    getRules : function(refreshCache){
14187                 if(rules == null || refreshCache){
14188                         rules = {};
14189                         var ds = doc.styleSheets;
14190                         for(var i =0, len = ds.length; i < len; i++){
14191                             try{
14192                         this.cacheStyleSheet(ds[i]);
14193                     }catch(e){} 
14194                 }
14195                 }
14196                 return rules;
14197         },
14198         
14199         /**
14200     * Gets an an individual CSS rule by selector(s)
14201     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14202     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14203     * @return {CSSRule} The CSS rule or null if one is not found
14204     */
14205    getRule : function(selector, refreshCache){
14206                 var rs = this.getRules(refreshCache);
14207                 if(!(selector instanceof Array)){
14208                     return rs[selector];
14209                 }
14210                 for(var i = 0; i < selector.length; i++){
14211                         if(rs[selector[i]]){
14212                                 return rs[selector[i]];
14213                         }
14214                 }
14215                 return null;
14216         },
14217         
14218         
14219         /**
14220     * Updates a rule property
14221     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14222     * @param {String} property The css property
14223     * @param {String} value The new value for the property
14224     * @return {Boolean} true If a rule was found and updated
14225     */
14226    updateRule : function(selector, property, value){
14227                 if(!(selector instanceof Array)){
14228                         var rule = this.getRule(selector);
14229                         if(rule){
14230                                 rule.style[property.replace(camelRe, camelFn)] = value;
14231                                 return true;
14232                         }
14233                 }else{
14234                         for(var i = 0; i < selector.length; i++){
14235                                 if(this.updateRule(selector[i], property, value)){
14236                                         return true;
14237                                 }
14238                         }
14239                 }
14240                 return false;
14241         }
14242    };   
14243 }();/*
14244  * Based on:
14245  * Ext JS Library 1.1.1
14246  * Copyright(c) 2006-2007, Ext JS, LLC.
14247  *
14248  * Originally Released Under LGPL - original licence link has changed is not relivant.
14249  *
14250  * Fork - LGPL
14251  * <script type="text/javascript">
14252  */
14253
14254  
14255
14256 /**
14257  * @class Roo.util.ClickRepeater
14258  * @extends Roo.util.Observable
14259  * 
14260  * A wrapper class which can be applied to any element. Fires a "click" event while the
14261  * mouse is pressed. The interval between firings may be specified in the config but
14262  * defaults to 10 milliseconds.
14263  * 
14264  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14265  * 
14266  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14267  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14268  * Similar to an autorepeat key delay.
14269  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14270  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14271  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14272  *           "interval" and "delay" are ignored. "immediate" is honored.
14273  * @cfg {Boolean} preventDefault True to prevent the default click event
14274  * @cfg {Boolean} stopDefault True to stop the default click event
14275  * 
14276  * @history
14277  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14278  *     2007-02-02 jvs Renamed to ClickRepeater
14279  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14280  *
14281  *  @constructor
14282  * @param {String/HTMLElement/Element} el The element to listen on
14283  * @param {Object} config
14284  **/
14285 Roo.util.ClickRepeater = function(el, config)
14286 {
14287     this.el = Roo.get(el);
14288     this.el.unselectable();
14289
14290     Roo.apply(this, config);
14291
14292     this.addEvents({
14293     /**
14294      * @event mousedown
14295      * Fires when the mouse button is depressed.
14296      * @param {Roo.util.ClickRepeater} this
14297      */
14298         "mousedown" : true,
14299     /**
14300      * @event click
14301      * Fires on a specified interval during the time the element is pressed.
14302      * @param {Roo.util.ClickRepeater} this
14303      */
14304         "click" : true,
14305     /**
14306      * @event mouseup
14307      * Fires when the mouse key is released.
14308      * @param {Roo.util.ClickRepeater} this
14309      */
14310         "mouseup" : true
14311     });
14312
14313     this.el.on("mousedown", this.handleMouseDown, this);
14314     if(this.preventDefault || this.stopDefault){
14315         this.el.on("click", function(e){
14316             if(this.preventDefault){
14317                 e.preventDefault();
14318             }
14319             if(this.stopDefault){
14320                 e.stopEvent();
14321             }
14322         }, this);
14323     }
14324
14325     // allow inline handler
14326     if(this.handler){
14327         this.on("click", this.handler,  this.scope || this);
14328     }
14329
14330     Roo.util.ClickRepeater.superclass.constructor.call(this);
14331 };
14332
14333 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14334     interval : 20,
14335     delay: 250,
14336     preventDefault : true,
14337     stopDefault : false,
14338     timer : 0,
14339
14340     // private
14341     handleMouseDown : function(){
14342         clearTimeout(this.timer);
14343         this.el.blur();
14344         if(this.pressClass){
14345             this.el.addClass(this.pressClass);
14346         }
14347         this.mousedownTime = new Date();
14348
14349         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14350         this.el.on("mouseout", this.handleMouseOut, this);
14351
14352         this.fireEvent("mousedown", this);
14353         this.fireEvent("click", this);
14354         
14355         this.timer = this.click.defer(this.delay || this.interval, this);
14356     },
14357
14358     // private
14359     click : function(){
14360         this.fireEvent("click", this);
14361         this.timer = this.click.defer(this.getInterval(), this);
14362     },
14363
14364     // private
14365     getInterval: function(){
14366         if(!this.accelerate){
14367             return this.interval;
14368         }
14369         var pressTime = this.mousedownTime.getElapsed();
14370         if(pressTime < 500){
14371             return 400;
14372         }else if(pressTime < 1700){
14373             return 320;
14374         }else if(pressTime < 2600){
14375             return 250;
14376         }else if(pressTime < 3500){
14377             return 180;
14378         }else if(pressTime < 4400){
14379             return 140;
14380         }else if(pressTime < 5300){
14381             return 80;
14382         }else if(pressTime < 6200){
14383             return 50;
14384         }else{
14385             return 10;
14386         }
14387     },
14388
14389     // private
14390     handleMouseOut : function(){
14391         clearTimeout(this.timer);
14392         if(this.pressClass){
14393             this.el.removeClass(this.pressClass);
14394         }
14395         this.el.on("mouseover", this.handleMouseReturn, this);
14396     },
14397
14398     // private
14399     handleMouseReturn : function(){
14400         this.el.un("mouseover", this.handleMouseReturn);
14401         if(this.pressClass){
14402             this.el.addClass(this.pressClass);
14403         }
14404         this.click();
14405     },
14406
14407     // private
14408     handleMouseUp : function(){
14409         clearTimeout(this.timer);
14410         this.el.un("mouseover", this.handleMouseReturn);
14411         this.el.un("mouseout", this.handleMouseOut);
14412         Roo.get(document).un("mouseup", this.handleMouseUp);
14413         this.el.removeClass(this.pressClass);
14414         this.fireEvent("mouseup", this);
14415     }
14416 });/*
14417  * Based on:
14418  * Ext JS Library 1.1.1
14419  * Copyright(c) 2006-2007, Ext JS, LLC.
14420  *
14421  * Originally Released Under LGPL - original licence link has changed is not relivant.
14422  *
14423  * Fork - LGPL
14424  * <script type="text/javascript">
14425  */
14426
14427  
14428 /**
14429  * @class Roo.KeyNav
14430  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14431  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14432  * way to implement custom navigation schemes for any UI component.</p>
14433  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14434  * pageUp, pageDown, del, home, end.  Usage:</p>
14435  <pre><code>
14436 var nav = new Roo.KeyNav("my-element", {
14437     "left" : function(e){
14438         this.moveLeft(e.ctrlKey);
14439     },
14440     "right" : function(e){
14441         this.moveRight(e.ctrlKey);
14442     },
14443     "enter" : function(e){
14444         this.save();
14445     },
14446     scope : this
14447 });
14448 </code></pre>
14449  * @constructor
14450  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14451  * @param {Object} config The config
14452  */
14453 Roo.KeyNav = function(el, config){
14454     this.el = Roo.get(el);
14455     Roo.apply(this, config);
14456     if(!this.disabled){
14457         this.disabled = true;
14458         this.enable();
14459     }
14460 };
14461
14462 Roo.KeyNav.prototype = {
14463     /**
14464      * @cfg {Boolean} disabled
14465      * True to disable this KeyNav instance (defaults to false)
14466      */
14467     disabled : false,
14468     /**
14469      * @cfg {String} defaultEventAction
14470      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14471      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14472      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14473      */
14474     defaultEventAction: "stopEvent",
14475     /**
14476      * @cfg {Boolean} forceKeyDown
14477      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14478      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14479      * handle keydown instead of keypress.
14480      */
14481     forceKeyDown : false,
14482
14483     // private
14484     prepareEvent : function(e){
14485         var k = e.getKey();
14486         var h = this.keyToHandler[k];
14487         //if(h && this[h]){
14488         //    e.stopPropagation();
14489         //}
14490         if(Roo.isSafari && h && k >= 37 && k <= 40){
14491             e.stopEvent();
14492         }
14493     },
14494
14495     // private
14496     relay : function(e){
14497         var k = e.getKey();
14498         var h = this.keyToHandler[k];
14499         if(h && this[h]){
14500             if(this.doRelay(e, this[h], h) !== true){
14501                 e[this.defaultEventAction]();
14502             }
14503         }
14504     },
14505
14506     // private
14507     doRelay : function(e, h, hname){
14508         return h.call(this.scope || this, e);
14509     },
14510
14511     // possible handlers
14512     enter : false,
14513     left : false,
14514     right : false,
14515     up : false,
14516     down : false,
14517     tab : false,
14518     esc : false,
14519     pageUp : false,
14520     pageDown : false,
14521     del : false,
14522     home : false,
14523     end : false,
14524
14525     // quick lookup hash
14526     keyToHandler : {
14527         37 : "left",
14528         39 : "right",
14529         38 : "up",
14530         40 : "down",
14531         33 : "pageUp",
14532         34 : "pageDown",
14533         46 : "del",
14534         36 : "home",
14535         35 : "end",
14536         13 : "enter",
14537         27 : "esc",
14538         9  : "tab"
14539     },
14540
14541         /**
14542          * Enable this KeyNav
14543          */
14544         enable: function(){
14545                 if(this.disabled){
14546             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14547             // the EventObject will normalize Safari automatically
14548             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14549                 this.el.on("keydown", this.relay,  this);
14550             }else{
14551                 this.el.on("keydown", this.prepareEvent,  this);
14552                 this.el.on("keypress", this.relay,  this);
14553             }
14554                     this.disabled = false;
14555                 }
14556         },
14557
14558         /**
14559          * Disable this KeyNav
14560          */
14561         disable: function(){
14562                 if(!this.disabled){
14563                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14564                 this.el.un("keydown", this.relay);
14565             }else{
14566                 this.el.un("keydown", this.prepareEvent);
14567                 this.el.un("keypress", this.relay);
14568             }
14569                     this.disabled = true;
14570                 }
14571         }
14572 };/*
14573  * Based on:
14574  * Ext JS Library 1.1.1
14575  * Copyright(c) 2006-2007, Ext JS, LLC.
14576  *
14577  * Originally Released Under LGPL - original licence link has changed is not relivant.
14578  *
14579  * Fork - LGPL
14580  * <script type="text/javascript">
14581  */
14582
14583  
14584 /**
14585  * @class Roo.KeyMap
14586  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14587  * The constructor accepts the same config object as defined by {@link #addBinding}.
14588  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14589  * combination it will call the function with this signature (if the match is a multi-key
14590  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14591  * A KeyMap can also handle a string representation of keys.<br />
14592  * Usage:
14593  <pre><code>
14594 // map one key by key code
14595 var map = new Roo.KeyMap("my-element", {
14596     key: 13, // or Roo.EventObject.ENTER
14597     fn: myHandler,
14598     scope: myObject
14599 });
14600
14601 // map multiple keys to one action by string
14602 var map = new Roo.KeyMap("my-element", {
14603     key: "a\r\n\t",
14604     fn: myHandler,
14605     scope: myObject
14606 });
14607
14608 // map multiple keys to multiple actions by strings and array of codes
14609 var map = new Roo.KeyMap("my-element", [
14610     {
14611         key: [10,13],
14612         fn: function(){ alert("Return was pressed"); }
14613     }, {
14614         key: "abc",
14615         fn: function(){ alert('a, b or c was pressed'); }
14616     }, {
14617         key: "\t",
14618         ctrl:true,
14619         shift:true,
14620         fn: function(){ alert('Control + shift + tab was pressed.'); }
14621     }
14622 ]);
14623 </code></pre>
14624  * <b>Note: A KeyMap starts enabled</b>
14625  * @constructor
14626  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14627  * @param {Object} config The config (see {@link #addBinding})
14628  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14629  */
14630 Roo.KeyMap = function(el, config, eventName){
14631     this.el  = Roo.get(el);
14632     this.eventName = eventName || "keydown";
14633     this.bindings = [];
14634     if(config){
14635         this.addBinding(config);
14636     }
14637     this.enable();
14638 };
14639
14640 Roo.KeyMap.prototype = {
14641     /**
14642      * True to stop the event from bubbling and prevent the default browser action if the
14643      * key was handled by the KeyMap (defaults to false)
14644      * @type Boolean
14645      */
14646     stopEvent : false,
14647
14648     /**
14649      * Add a new binding to this KeyMap. The following config object properties are supported:
14650      * <pre>
14651 Property    Type             Description
14652 ----------  ---------------  ----------------------------------------------------------------------
14653 key         String/Array     A single keycode or an array of keycodes to handle
14654 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14655 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14656 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14657 fn          Function         The function to call when KeyMap finds the expected key combination
14658 scope       Object           The scope of the callback function
14659 </pre>
14660      *
14661      * Usage:
14662      * <pre><code>
14663 // Create a KeyMap
14664 var map = new Roo.KeyMap(document, {
14665     key: Roo.EventObject.ENTER,
14666     fn: handleKey,
14667     scope: this
14668 });
14669
14670 //Add a new binding to the existing KeyMap later
14671 map.addBinding({
14672     key: 'abc',
14673     shift: true,
14674     fn: handleKey,
14675     scope: this
14676 });
14677 </code></pre>
14678      * @param {Object/Array} config A single KeyMap config or an array of configs
14679      */
14680         addBinding : function(config){
14681         if(config instanceof Array){
14682             for(var i = 0, len = config.length; i < len; i++){
14683                 this.addBinding(config[i]);
14684             }
14685             return;
14686         }
14687         var keyCode = config.key,
14688             shift = config.shift, 
14689             ctrl = config.ctrl, 
14690             alt = config.alt,
14691             fn = config.fn,
14692             scope = config.scope;
14693         if(typeof keyCode == "string"){
14694             var ks = [];
14695             var keyString = keyCode.toUpperCase();
14696             for(var j = 0, len = keyString.length; j < len; j++){
14697                 ks.push(keyString.charCodeAt(j));
14698             }
14699             keyCode = ks;
14700         }
14701         var keyArray = keyCode instanceof Array;
14702         var handler = function(e){
14703             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14704                 var k = e.getKey();
14705                 if(keyArray){
14706                     for(var i = 0, len = keyCode.length; i < len; i++){
14707                         if(keyCode[i] == k){
14708                           if(this.stopEvent){
14709                               e.stopEvent();
14710                           }
14711                           fn.call(scope || window, k, e);
14712                           return;
14713                         }
14714                     }
14715                 }else{
14716                     if(k == keyCode){
14717                         if(this.stopEvent){
14718                            e.stopEvent();
14719                         }
14720                         fn.call(scope || window, k, e);
14721                     }
14722                 }
14723             }
14724         };
14725         this.bindings.push(handler);  
14726         },
14727
14728     /**
14729      * Shorthand for adding a single key listener
14730      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14731      * following options:
14732      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14733      * @param {Function} fn The function to call
14734      * @param {Object} scope (optional) The scope of the function
14735      */
14736     on : function(key, fn, scope){
14737         var keyCode, shift, ctrl, alt;
14738         if(typeof key == "object" && !(key instanceof Array)){
14739             keyCode = key.key;
14740             shift = key.shift;
14741             ctrl = key.ctrl;
14742             alt = key.alt;
14743         }else{
14744             keyCode = key;
14745         }
14746         this.addBinding({
14747             key: keyCode,
14748             shift: shift,
14749             ctrl: ctrl,
14750             alt: alt,
14751             fn: fn,
14752             scope: scope
14753         })
14754     },
14755
14756     // private
14757     handleKeyDown : function(e){
14758             if(this.enabled){ //just in case
14759             var b = this.bindings;
14760             for(var i = 0, len = b.length; i < len; i++){
14761                 b[i].call(this, e);
14762             }
14763             }
14764         },
14765         
14766         /**
14767          * Returns true if this KeyMap is enabled
14768          * @return {Boolean} 
14769          */
14770         isEnabled : function(){
14771             return this.enabled;  
14772         },
14773         
14774         /**
14775          * Enables this KeyMap
14776          */
14777         enable: function(){
14778                 if(!this.enabled){
14779                     this.el.on(this.eventName, this.handleKeyDown, this);
14780                     this.enabled = true;
14781                 }
14782         },
14783
14784         /**
14785          * Disable this KeyMap
14786          */
14787         disable: function(){
14788                 if(this.enabled){
14789                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14790                     this.enabled = false;
14791                 }
14792         }
14793 };/*
14794  * Based on:
14795  * Ext JS Library 1.1.1
14796  * Copyright(c) 2006-2007, Ext JS, LLC.
14797  *
14798  * Originally Released Under LGPL - original licence link has changed is not relivant.
14799  *
14800  * Fork - LGPL
14801  * <script type="text/javascript">
14802  */
14803
14804  
14805 /**
14806  * @class Roo.util.TextMetrics
14807  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14808  * wide, in pixels, a given block of text will be.
14809  * @singleton
14810  */
14811 Roo.util.TextMetrics = function(){
14812     var shared;
14813     return {
14814         /**
14815          * Measures the size of the specified text
14816          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14817          * that can affect the size of the rendered text
14818          * @param {String} text The text to measure
14819          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14820          * in order to accurately measure the text height
14821          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14822          */
14823         measure : function(el, text, fixedWidth){
14824             if(!shared){
14825                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14826             }
14827             shared.bind(el);
14828             shared.setFixedWidth(fixedWidth || 'auto');
14829             return shared.getSize(text);
14830         },
14831
14832         /**
14833          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14834          * the overhead of multiple calls to initialize the style properties on each measurement.
14835          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14836          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14837          * in order to accurately measure the text height
14838          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14839          */
14840         createInstance : function(el, fixedWidth){
14841             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14842         }
14843     };
14844 }();
14845
14846  
14847
14848 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14849     var ml = new Roo.Element(document.createElement('div'));
14850     document.body.appendChild(ml.dom);
14851     ml.position('absolute');
14852     ml.setLeftTop(-1000, -1000);
14853     ml.hide();
14854
14855     if(fixedWidth){
14856         ml.setWidth(fixedWidth);
14857     }
14858      
14859     var instance = {
14860         /**
14861          * Returns the size of the specified text based on the internal element's style and width properties
14862          * @memberOf Roo.util.TextMetrics.Instance#
14863          * @param {String} text The text to measure
14864          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14865          */
14866         getSize : function(text){
14867             ml.update(text);
14868             var s = ml.getSize();
14869             ml.update('');
14870             return s;
14871         },
14872
14873         /**
14874          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14875          * that can affect the size of the rendered text
14876          * @memberOf Roo.util.TextMetrics.Instance#
14877          * @param {String/HTMLElement} el The element, dom node or id
14878          */
14879         bind : function(el){
14880             ml.setStyle(
14881                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14882             );
14883         },
14884
14885         /**
14886          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14887          * to set a fixed width in order to accurately measure the text height.
14888          * @memberOf Roo.util.TextMetrics.Instance#
14889          * @param {Number} width The width to set on the element
14890          */
14891         setFixedWidth : function(width){
14892             ml.setWidth(width);
14893         },
14894
14895         /**
14896          * Returns the measured width of the specified text
14897          * @memberOf Roo.util.TextMetrics.Instance#
14898          * @param {String} text The text to measure
14899          * @return {Number} width The width in pixels
14900          */
14901         getWidth : function(text){
14902             ml.dom.style.width = 'auto';
14903             return this.getSize(text).width;
14904         },
14905
14906         /**
14907          * Returns the measured height of the specified text.  For multiline text, be sure to call
14908          * {@link #setFixedWidth} if necessary.
14909          * @memberOf Roo.util.TextMetrics.Instance#
14910          * @param {String} text The text to measure
14911          * @return {Number} height The height in pixels
14912          */
14913         getHeight : function(text){
14914             return this.getSize(text).height;
14915         }
14916     };
14917
14918     instance.bind(bindTo);
14919
14920     return instance;
14921 };
14922
14923 // backwards compat
14924 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14925  * Based on:
14926  * Ext JS Library 1.1.1
14927  * Copyright(c) 2006-2007, Ext JS, LLC.
14928  *
14929  * Originally Released Under LGPL - original licence link has changed is not relivant.
14930  *
14931  * Fork - LGPL
14932  * <script type="text/javascript">
14933  */
14934
14935 /**
14936  * @class Roo.state.Provider
14937  * Abstract base class for state provider implementations. This class provides methods
14938  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14939  * Provider interface.
14940  */
14941 Roo.state.Provider = function(){
14942     /**
14943      * @event statechange
14944      * Fires when a state change occurs.
14945      * @param {Provider} this This state provider
14946      * @param {String} key The state key which was changed
14947      * @param {String} value The encoded value for the state
14948      */
14949     this.addEvents({
14950         "statechange": true
14951     });
14952     this.state = {};
14953     Roo.state.Provider.superclass.constructor.call(this);
14954 };
14955 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14956     /**
14957      * Returns the current value for a key
14958      * @param {String} name The key name
14959      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14960      * @return {Mixed} The state data
14961      */
14962     get : function(name, defaultValue){
14963         return typeof this.state[name] == "undefined" ?
14964             defaultValue : this.state[name];
14965     },
14966     
14967     /**
14968      * Clears a value from the state
14969      * @param {String} name The key name
14970      */
14971     clear : function(name){
14972         delete this.state[name];
14973         this.fireEvent("statechange", this, name, null);
14974     },
14975     
14976     /**
14977      * Sets the value for a key
14978      * @param {String} name The key name
14979      * @param {Mixed} value The value to set
14980      */
14981     set : function(name, value){
14982         this.state[name] = value;
14983         this.fireEvent("statechange", this, name, value);
14984     },
14985     
14986     /**
14987      * Decodes a string previously encoded with {@link #encodeValue}.
14988      * @param {String} value The value to decode
14989      * @return {Mixed} The decoded value
14990      */
14991     decodeValue : function(cookie){
14992         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14993         var matches = re.exec(unescape(cookie));
14994         if(!matches || !matches[1]) {
14995             return; // non state cookie
14996         }
14997         var type = matches[1];
14998         var v = matches[2];
14999         switch(type){
15000             case "n":
15001                 return parseFloat(v);
15002             case "d":
15003                 return new Date(Date.parse(v));
15004             case "b":
15005                 return (v == "1");
15006             case "a":
15007                 var all = [];
15008                 var values = v.split("^");
15009                 for(var i = 0, len = values.length; i < len; i++){
15010                     all.push(this.decodeValue(values[i]));
15011                 }
15012                 return all;
15013            case "o":
15014                 var all = {};
15015                 var values = v.split("^");
15016                 for(var i = 0, len = values.length; i < len; i++){
15017                     var kv = values[i].split("=");
15018                     all[kv[0]] = this.decodeValue(kv[1]);
15019                 }
15020                 return all;
15021            default:
15022                 return v;
15023         }
15024     },
15025     
15026     /**
15027      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15028      * @param {Mixed} value The value to encode
15029      * @return {String} The encoded value
15030      */
15031     encodeValue : function(v){
15032         var enc;
15033         if(typeof v == "number"){
15034             enc = "n:" + v;
15035         }else if(typeof v == "boolean"){
15036             enc = "b:" + (v ? "1" : "0");
15037         }else if(v instanceof Date){
15038             enc = "d:" + v.toGMTString();
15039         }else if(v instanceof Array){
15040             var flat = "";
15041             for(var i = 0, len = v.length; i < len; i++){
15042                 flat += this.encodeValue(v[i]);
15043                 if(i != len-1) {
15044                     flat += "^";
15045                 }
15046             }
15047             enc = "a:" + flat;
15048         }else if(typeof v == "object"){
15049             var flat = "";
15050             for(var key in v){
15051                 if(typeof v[key] != "function"){
15052                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15053                 }
15054             }
15055             enc = "o:" + flat.substring(0, flat.length-1);
15056         }else{
15057             enc = "s:" + v;
15058         }
15059         return escape(enc);        
15060     }
15061 });
15062
15063 /*
15064  * Based on:
15065  * Ext JS Library 1.1.1
15066  * Copyright(c) 2006-2007, Ext JS, LLC.
15067  *
15068  * Originally Released Under LGPL - original licence link has changed is not relivant.
15069  *
15070  * Fork - LGPL
15071  * <script type="text/javascript">
15072  */
15073 /**
15074  * @class Roo.state.Manager
15075  * This is the global state manager. By default all components that are "state aware" check this class
15076  * for state information if you don't pass them a custom state provider. In order for this class
15077  * to be useful, it must be initialized with a provider when your application initializes.
15078  <pre><code>
15079 // in your initialization function
15080 init : function(){
15081    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15082    ...
15083    // supposed you have a {@link Roo.BorderLayout}
15084    var layout = new Roo.BorderLayout(...);
15085    layout.restoreState();
15086    // or a {Roo.BasicDialog}
15087    var dialog = new Roo.BasicDialog(...);
15088    dialog.restoreState();
15089  </code></pre>
15090  * @singleton
15091  */
15092 Roo.state.Manager = function(){
15093     var provider = new Roo.state.Provider();
15094     
15095     return {
15096         /**
15097          * Configures the default state provider for your application
15098          * @param {Provider} stateProvider The state provider to set
15099          */
15100         setProvider : function(stateProvider){
15101             provider = stateProvider;
15102         },
15103         
15104         /**
15105          * Returns the current value for a key
15106          * @param {String} name The key name
15107          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15108          * @return {Mixed} The state data
15109          */
15110         get : function(key, defaultValue){
15111             return provider.get(key, defaultValue);
15112         },
15113         
15114         /**
15115          * Sets the value for a key
15116          * @param {String} name The key name
15117          * @param {Mixed} value The state data
15118          */
15119          set : function(key, value){
15120             provider.set(key, value);
15121         },
15122         
15123         /**
15124          * Clears a value from the state
15125          * @param {String} name The key name
15126          */
15127         clear : function(key){
15128             provider.clear(key);
15129         },
15130         
15131         /**
15132          * Gets the currently configured state provider
15133          * @return {Provider} The state provider
15134          */
15135         getProvider : function(){
15136             return provider;
15137         }
15138     };
15139 }();
15140 /*
15141  * Based on:
15142  * Ext JS Library 1.1.1
15143  * Copyright(c) 2006-2007, Ext JS, LLC.
15144  *
15145  * Originally Released Under LGPL - original licence link has changed is not relivant.
15146  *
15147  * Fork - LGPL
15148  * <script type="text/javascript">
15149  */
15150 /**
15151  * @class Roo.state.CookieProvider
15152  * @extends Roo.state.Provider
15153  * The default Provider implementation which saves state via cookies.
15154  * <br />Usage:
15155  <pre><code>
15156    var cp = new Roo.state.CookieProvider({
15157        path: "/cgi-bin/",
15158        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15159        domain: "roojs.com"
15160    })
15161    Roo.state.Manager.setProvider(cp);
15162  </code></pre>
15163  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15164  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15165  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15166  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15167  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15168  * domain the page is running on including the 'www' like 'www.roojs.com')
15169  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15170  * @constructor
15171  * Create a new CookieProvider
15172  * @param {Object} config The configuration object
15173  */
15174 Roo.state.CookieProvider = function(config){
15175     Roo.state.CookieProvider.superclass.constructor.call(this);
15176     this.path = "/";
15177     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15178     this.domain = null;
15179     this.secure = false;
15180     Roo.apply(this, config);
15181     this.state = this.readCookies();
15182 };
15183
15184 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15185     // private
15186     set : function(name, value){
15187         if(typeof value == "undefined" || value === null){
15188             this.clear(name);
15189             return;
15190         }
15191         this.setCookie(name, value);
15192         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15193     },
15194
15195     // private
15196     clear : function(name){
15197         this.clearCookie(name);
15198         Roo.state.CookieProvider.superclass.clear.call(this, name);
15199     },
15200
15201     // private
15202     readCookies : function(){
15203         var cookies = {};
15204         var c = document.cookie + ";";
15205         var re = /\s?(.*?)=(.*?);/g;
15206         var matches;
15207         while((matches = re.exec(c)) != null){
15208             var name = matches[1];
15209             var value = matches[2];
15210             if(name && name.substring(0,3) == "ys-"){
15211                 cookies[name.substr(3)] = this.decodeValue(value);
15212             }
15213         }
15214         return cookies;
15215     },
15216
15217     // private
15218     setCookie : function(name, value){
15219         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15220            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15221            ((this.path == null) ? "" : ("; path=" + this.path)) +
15222            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15223            ((this.secure == true) ? "; secure" : "");
15224     },
15225
15226     // private
15227     clearCookie : function(name){
15228         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15229            ((this.path == null) ? "" : ("; path=" + this.path)) +
15230            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15231            ((this.secure == true) ? "; secure" : "");
15232     }
15233 });/*
15234  * Based on:
15235  * Ext JS Library 1.1.1
15236  * Copyright(c) 2006-2007, Ext JS, LLC.
15237  *
15238  * Originally Released Under LGPL - original licence link has changed is not relivant.
15239  *
15240  * Fork - LGPL
15241  * <script type="text/javascript">
15242  */
15243  
15244
15245 /**
15246  * @class Roo.ComponentMgr
15247  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15248  * @singleton
15249  */
15250 Roo.ComponentMgr = function(){
15251     var all = new Roo.util.MixedCollection();
15252
15253     return {
15254         /**
15255          * Registers a component.
15256          * @param {Roo.Component} c The component
15257          */
15258         register : function(c){
15259             all.add(c);
15260         },
15261
15262         /**
15263          * Unregisters a component.
15264          * @param {Roo.Component} c The component
15265          */
15266         unregister : function(c){
15267             all.remove(c);
15268         },
15269
15270         /**
15271          * Returns a component by id
15272          * @param {String} id The component id
15273          */
15274         get : function(id){
15275             return all.get(id);
15276         },
15277
15278         /**
15279          * Registers a function that will be called when a specified component is added to ComponentMgr
15280          * @param {String} id The component id
15281          * @param {Funtction} fn The callback function
15282          * @param {Object} scope The scope of the callback
15283          */
15284         onAvailable : function(id, fn, scope){
15285             all.on("add", function(index, o){
15286                 if(o.id == id){
15287                     fn.call(scope || o, o);
15288                     all.un("add", fn, scope);
15289                 }
15290             });
15291         }
15292     };
15293 }();/*
15294  * Based on:
15295  * Ext JS Library 1.1.1
15296  * Copyright(c) 2006-2007, Ext JS, LLC.
15297  *
15298  * Originally Released Under LGPL - original licence link has changed is not relivant.
15299  *
15300  * Fork - LGPL
15301  * <script type="text/javascript">
15302  */
15303  
15304 /**
15305  * @class Roo.Component
15306  * @extends Roo.util.Observable
15307  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15308  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15309  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15310  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15311  * All visual components (widgets) that require rendering into a layout should subclass Component.
15312  * @constructor
15313  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15314  * element and its id used as the component id.  If a string is passed, it is assumed to be the id of an existing element
15315  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15316  */
15317 Roo.Component = function(config){
15318     config = config || {};
15319     if(config.tagName || config.dom || typeof config == "string"){ // element object
15320         config = {el: config, id: config.id || config};
15321     }
15322     this.initialConfig = config;
15323
15324     Roo.apply(this, config);
15325     this.addEvents({
15326         /**
15327          * @event disable
15328          * Fires after the component is disabled.
15329              * @param {Roo.Component} this
15330              */
15331         disable : true,
15332         /**
15333          * @event enable
15334          * Fires after the component is enabled.
15335              * @param {Roo.Component} this
15336              */
15337         enable : true,
15338         /**
15339          * @event beforeshow
15340          * Fires before the component is shown.  Return false to stop the show.
15341              * @param {Roo.Component} this
15342              */
15343         beforeshow : true,
15344         /**
15345          * @event show
15346          * Fires after the component is shown.
15347              * @param {Roo.Component} this
15348              */
15349         show : true,
15350         /**
15351          * @event beforehide
15352          * Fires before the component is hidden. Return false to stop the hide.
15353              * @param {Roo.Component} this
15354              */
15355         beforehide : true,
15356         /**
15357          * @event hide
15358          * Fires after the component is hidden.
15359              * @param {Roo.Component} this
15360              */
15361         hide : true,
15362         /**
15363          * @event beforerender
15364          * Fires before the component is rendered. Return false to stop the render.
15365              * @param {Roo.Component} this
15366              */
15367         beforerender : true,
15368         /**
15369          * @event render
15370          * Fires after the component is rendered.
15371              * @param {Roo.Component} this
15372              */
15373         render : true,
15374         /**
15375          * @event beforedestroy
15376          * Fires before the component is destroyed. Return false to stop the destroy.
15377              * @param {Roo.Component} this
15378              */
15379         beforedestroy : true,
15380         /**
15381          * @event destroy
15382          * Fires after the component is destroyed.
15383              * @param {Roo.Component} this
15384              */
15385         destroy : true
15386     });
15387     if(!this.id){
15388         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15389     }
15390     Roo.ComponentMgr.register(this);
15391     Roo.Component.superclass.constructor.call(this);
15392     this.initComponent();
15393     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15394         this.render(this.renderTo);
15395         delete this.renderTo;
15396     }
15397 };
15398
15399 /** @private */
15400 Roo.Component.AUTO_ID = 1000;
15401
15402 Roo.extend(Roo.Component, Roo.util.Observable, {
15403     /**
15404      * @scope Roo.Component.prototype
15405      * @type {Boolean}
15406      * true if this component is hidden. Read-only.
15407      */
15408     hidden : false,
15409     /**
15410      * @type {Boolean}
15411      * true if this component is disabled. Read-only.
15412      */
15413     disabled : false,
15414     /**
15415      * @type {Boolean}
15416      * true if this component has been rendered. Read-only.
15417      */
15418     rendered : false,
15419     
15420     /** @cfg {String} disableClass
15421      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15422      */
15423     disabledClass : "x-item-disabled",
15424         /** @cfg {Boolean} allowDomMove
15425          * Whether the component can move the Dom node when rendering (defaults to true).
15426          */
15427     allowDomMove : true,
15428     /** @cfg {String} hideMode (display|visibility)
15429      * How this component should hidden. Supported values are
15430      * "visibility" (css visibility), "offsets" (negative offset position) and
15431      * "display" (css display) - defaults to "display".
15432      */
15433     hideMode: 'display',
15434
15435     /** @private */
15436     ctype : "Roo.Component",
15437
15438     /**
15439      * @cfg {String} actionMode 
15440      * which property holds the element that used for  hide() / show() / disable() / enable()
15441      * default is 'el' 
15442      */
15443     actionMode : "el",
15444
15445     /** @private */
15446     getActionEl : function(){
15447         return this[this.actionMode];
15448     },
15449
15450     initComponent : Roo.emptyFn,
15451     /**
15452      * If this is a lazy rendering component, render it to its container element.
15453      * @param {String/HTMLElement/Element} container (optional) The element this component should be rendered into. If it is being applied to existing markup, this should be left off.
15454      */
15455     render : function(container, position){
15456         
15457         if(this.rendered){
15458             return this;
15459         }
15460         
15461         if(this.fireEvent("beforerender", this) === false){
15462             return false;
15463         }
15464         
15465         if(!container && this.el){
15466             this.el = Roo.get(this.el);
15467             container = this.el.dom.parentNode;
15468             this.allowDomMove = false;
15469         }
15470         this.container = Roo.get(container);
15471         this.rendered = true;
15472         if(position !== undefined){
15473             if(typeof position == 'number'){
15474                 position = this.container.dom.childNodes[position];
15475             }else{
15476                 position = Roo.getDom(position);
15477             }
15478         }
15479         this.onRender(this.container, position || null);
15480         if(this.cls){
15481             this.el.addClass(this.cls);
15482             delete this.cls;
15483         }
15484         if(this.style){
15485             this.el.applyStyles(this.style);
15486             delete this.style;
15487         }
15488         this.fireEvent("render", this);
15489         this.afterRender(this.container);
15490         if(this.hidden){
15491             this.hide();
15492         }
15493         if(this.disabled){
15494             this.disable();
15495         }
15496
15497         return this;
15498         
15499     },
15500
15501     /** @private */
15502     // default function is not really useful
15503     onRender : function(ct, position){
15504         if(this.el){
15505             this.el = Roo.get(this.el);
15506             if(this.allowDomMove !== false){
15507                 ct.dom.insertBefore(this.el.dom, position);
15508             }
15509         }
15510     },
15511
15512     /** @private */
15513     getAutoCreate : function(){
15514         var cfg = typeof this.autoCreate == "object" ?
15515                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15516         if(this.id && !cfg.id){
15517             cfg.id = this.id;
15518         }
15519         return cfg;
15520     },
15521
15522     /** @private */
15523     afterRender : Roo.emptyFn,
15524
15525     /**
15526      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15527      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15528      */
15529     destroy : function(){
15530         if(this.fireEvent("beforedestroy", this) !== false){
15531             this.purgeListeners();
15532             this.beforeDestroy();
15533             if(this.rendered){
15534                 this.el.removeAllListeners();
15535                 this.el.remove();
15536                 if(this.actionMode == "container"){
15537                     this.container.remove();
15538                 }
15539             }
15540             this.onDestroy();
15541             Roo.ComponentMgr.unregister(this);
15542             this.fireEvent("destroy", this);
15543         }
15544     },
15545
15546         /** @private */
15547     beforeDestroy : function(){
15548
15549     },
15550
15551         /** @private */
15552         onDestroy : function(){
15553
15554     },
15555
15556     /**
15557      * Returns the underlying {@link Roo.Element}.
15558      * @return {Roo.Element} The element
15559      */
15560     getEl : function(){
15561         return this.el;
15562     },
15563
15564     /**
15565      * Returns the id of this component.
15566      * @return {String}
15567      */
15568     getId : function(){
15569         return this.id;
15570     },
15571
15572     /**
15573      * Try to focus this component.
15574      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15575      * @return {Roo.Component} this
15576      */
15577     focus : function(selectText){
15578         if(this.rendered){
15579             this.el.focus();
15580             if(selectText === true){
15581                 this.el.dom.select();
15582             }
15583         }
15584         return this;
15585     },
15586
15587     /** @private */
15588     blur : function(){
15589         if(this.rendered){
15590             this.el.blur();
15591         }
15592         return this;
15593     },
15594
15595     /**
15596      * Disable this component.
15597      * @return {Roo.Component} this
15598      */
15599     disable : function(){
15600         if(this.rendered){
15601             this.onDisable();
15602         }
15603         this.disabled = true;
15604         this.fireEvent("disable", this);
15605         return this;
15606     },
15607
15608         // private
15609     onDisable : function(){
15610         this.getActionEl().addClass(this.disabledClass);
15611         this.el.dom.disabled = true;
15612     },
15613
15614     /**
15615      * Enable this component.
15616      * @return {Roo.Component} this
15617      */
15618     enable : function(){
15619         if(this.rendered){
15620             this.onEnable();
15621         }
15622         this.disabled = false;
15623         this.fireEvent("enable", this);
15624         return this;
15625     },
15626
15627         // private
15628     onEnable : function(){
15629         this.getActionEl().removeClass(this.disabledClass);
15630         this.el.dom.disabled = false;
15631     },
15632
15633     /**
15634      * Convenience function for setting disabled/enabled by boolean.
15635      * @param {Boolean} disabled
15636      */
15637     setDisabled : function(disabled){
15638         this[disabled ? "disable" : "enable"]();
15639     },
15640
15641     /**
15642      * Show this component.
15643      * @return {Roo.Component} this
15644      */
15645     show: function(){
15646         if(this.fireEvent("beforeshow", this) !== false){
15647             this.hidden = false;
15648             if(this.rendered){
15649                 this.onShow();
15650             }
15651             this.fireEvent("show", this);
15652         }
15653         return this;
15654     },
15655
15656     // private
15657     onShow : function(){
15658         var ae = this.getActionEl();
15659         if(this.hideMode == 'visibility'){
15660             ae.dom.style.visibility = "visible";
15661         }else if(this.hideMode == 'offsets'){
15662             ae.removeClass('x-hidden');
15663         }else{
15664             ae.dom.style.display = "";
15665         }
15666     },
15667
15668     /**
15669      * Hide this component.
15670      * @return {Roo.Component} this
15671      */
15672     hide: function(){
15673         if(this.fireEvent("beforehide", this) !== false){
15674             this.hidden = true;
15675             if(this.rendered){
15676                 this.onHide();
15677             }
15678             this.fireEvent("hide", this);
15679         }
15680         return this;
15681     },
15682
15683     // private
15684     onHide : function(){
15685         var ae = this.getActionEl();
15686         if(this.hideMode == 'visibility'){
15687             ae.dom.style.visibility = "hidden";
15688         }else if(this.hideMode == 'offsets'){
15689             ae.addClass('x-hidden');
15690         }else{
15691             ae.dom.style.display = "none";
15692         }
15693     },
15694
15695     /**
15696      * Convenience function to hide or show this component by boolean.
15697      * @param {Boolean} visible True to show, false to hide
15698      * @return {Roo.Component} this
15699      */
15700     setVisible: function(visible){
15701         if(visible) {
15702             this.show();
15703         }else{
15704             this.hide();
15705         }
15706         return this;
15707     },
15708
15709     /**
15710      * Returns true if this component is visible.
15711      */
15712     isVisible : function(){
15713         return this.getActionEl().isVisible();
15714     },
15715
15716     cloneConfig : function(overrides){
15717         overrides = overrides || {};
15718         var id = overrides.id || Roo.id();
15719         var cfg = Roo.applyIf(overrides, this.initialConfig);
15720         cfg.id = id; // prevent dup id
15721         return new this.constructor(cfg);
15722     }
15723 });/*
15724  * Based on:
15725  * Ext JS Library 1.1.1
15726  * Copyright(c) 2006-2007, Ext JS, LLC.
15727  *
15728  * Originally Released Under LGPL - original licence link has changed is not relivant.
15729  *
15730  * Fork - LGPL
15731  * <script type="text/javascript">
15732  */
15733
15734 /**
15735  * @class Roo.BoxComponent
15736  * @extends Roo.Component
15737  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15738  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15739  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15740  * layout containers.
15741  * @constructor
15742  * @param {Roo.Element/String/Object} config The configuration options.
15743  */
15744 Roo.BoxComponent = function(config){
15745     Roo.Component.call(this, config);
15746     this.addEvents({
15747         /**
15748          * @event resize
15749          * Fires after the component is resized.
15750              * @param {Roo.Component} this
15751              * @param {Number} adjWidth The box-adjusted width that was set
15752              * @param {Number} adjHeight The box-adjusted height that was set
15753              * @param {Number} rawWidth The width that was originally specified
15754              * @param {Number} rawHeight The height that was originally specified
15755              */
15756         resize : true,
15757         /**
15758          * @event move
15759          * Fires after the component is moved.
15760              * @param {Roo.Component} this
15761              * @param {Number} x The new x position
15762              * @param {Number} y The new y position
15763              */
15764         move : true
15765     });
15766 };
15767
15768 Roo.extend(Roo.BoxComponent, Roo.Component, {
15769     // private, set in afterRender to signify that the component has been rendered
15770     boxReady : false,
15771     // private, used to defer height settings to subclasses
15772     deferHeight: false,
15773     /** @cfg {Number} width
15774      * width (optional) size of component
15775      */
15776      /** @cfg {Number} height
15777      * height (optional) size of component
15778      */
15779      
15780     /**
15781      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15782      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15783      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15784      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15785      * @return {Roo.BoxComponent} this
15786      */
15787     setSize : function(w, h){
15788         // support for standard size objects
15789         if(typeof w == 'object'){
15790             h = w.height;
15791             w = w.width;
15792         }
15793         // not rendered
15794         if(!this.boxReady){
15795             this.width = w;
15796             this.height = h;
15797             return this;
15798         }
15799
15800         // prevent recalcs when not needed
15801         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15802             return this;
15803         }
15804         this.lastSize = {width: w, height: h};
15805
15806         var adj = this.adjustSize(w, h);
15807         var aw = adj.width, ah = adj.height;
15808         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15809             var rz = this.getResizeEl();
15810             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15811                 rz.setSize(aw, ah);
15812             }else if(!this.deferHeight && ah !== undefined){
15813                 rz.setHeight(ah);
15814             }else if(aw !== undefined){
15815                 rz.setWidth(aw);
15816             }
15817             this.onResize(aw, ah, w, h);
15818             this.fireEvent('resize', this, aw, ah, w, h);
15819         }
15820         return this;
15821     },
15822
15823     /**
15824      * Gets the current size of the component's underlying element.
15825      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15826      */
15827     getSize : function(){
15828         return this.el.getSize();
15829     },
15830
15831     /**
15832      * Gets the current XY position of the component's underlying element.
15833      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15834      * @return {Array} The XY position of the element (e.g., [100, 200])
15835      */
15836     getPosition : function(local){
15837         if(local === true){
15838             return [this.el.getLeft(true), this.el.getTop(true)];
15839         }
15840         return this.xy || this.el.getXY();
15841     },
15842
15843     /**
15844      * Gets the current box measurements of the component's underlying element.
15845      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15846      * @returns {Object} box An object in the format {x, y, width, height}
15847      */
15848     getBox : function(local){
15849         var s = this.el.getSize();
15850         if(local){
15851             s.x = this.el.getLeft(true);
15852             s.y = this.el.getTop(true);
15853         }else{
15854             var xy = this.xy || this.el.getXY();
15855             s.x = xy[0];
15856             s.y = xy[1];
15857         }
15858         return s;
15859     },
15860
15861     /**
15862      * Sets the current box measurements of the component's underlying element.
15863      * @param {Object} box An object in the format {x, y, width, height}
15864      * @returns {Roo.BoxComponent} this
15865      */
15866     updateBox : function(box){
15867         this.setSize(box.width, box.height);
15868         this.setPagePosition(box.x, box.y);
15869         return this;
15870     },
15871
15872     // protected
15873     getResizeEl : function(){
15874         return this.resizeEl || this.el;
15875     },
15876
15877     // protected
15878     getPositionEl : function(){
15879         return this.positionEl || this.el;
15880     },
15881
15882     /**
15883      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15884      * This method fires the move event.
15885      * @param {Number} left The new left
15886      * @param {Number} top The new top
15887      * @returns {Roo.BoxComponent} this
15888      */
15889     setPosition : function(x, y){
15890         this.x = x;
15891         this.y = y;
15892         if(!this.boxReady){
15893             return this;
15894         }
15895         var adj = this.adjustPosition(x, y);
15896         var ax = adj.x, ay = adj.y;
15897
15898         var el = this.getPositionEl();
15899         if(ax !== undefined || ay !== undefined){
15900             if(ax !== undefined && ay !== undefined){
15901                 el.setLeftTop(ax, ay);
15902             }else if(ax !== undefined){
15903                 el.setLeft(ax);
15904             }else if(ay !== undefined){
15905                 el.setTop(ay);
15906             }
15907             this.onPosition(ax, ay);
15908             this.fireEvent('move', this, ax, ay);
15909         }
15910         return this;
15911     },
15912
15913     /**
15914      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15915      * This method fires the move event.
15916      * @param {Number} x The new x position
15917      * @param {Number} y The new y position
15918      * @returns {Roo.BoxComponent} this
15919      */
15920     setPagePosition : function(x, y){
15921         this.pageX = x;
15922         this.pageY = y;
15923         if(!this.boxReady){
15924             return;
15925         }
15926         if(x === undefined || y === undefined){ // cannot translate undefined points
15927             return;
15928         }
15929         var p = this.el.translatePoints(x, y);
15930         this.setPosition(p.left, p.top);
15931         return this;
15932     },
15933
15934     // private
15935     onRender : function(ct, position){
15936         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15937         if(this.resizeEl){
15938             this.resizeEl = Roo.get(this.resizeEl);
15939         }
15940         if(this.positionEl){
15941             this.positionEl = Roo.get(this.positionEl);
15942         }
15943     },
15944
15945     // private
15946     afterRender : function(){
15947         Roo.BoxComponent.superclass.afterRender.call(this);
15948         this.boxReady = true;
15949         this.setSize(this.width, this.height);
15950         if(this.x || this.y){
15951             this.setPosition(this.x, this.y);
15952         }
15953         if(this.pageX || this.pageY){
15954             this.setPagePosition(this.pageX, this.pageY);
15955         }
15956     },
15957
15958     /**
15959      * Force the component's size to recalculate based on the underlying element's current height and width.
15960      * @returns {Roo.BoxComponent} this
15961      */
15962     syncSize : function(){
15963         delete this.lastSize;
15964         this.setSize(this.el.getWidth(), this.el.getHeight());
15965         return this;
15966     },
15967
15968     /**
15969      * Called after the component is resized, this method is empty by default but can be implemented by any
15970      * subclass that needs to perform custom logic after a resize occurs.
15971      * @param {Number} adjWidth The box-adjusted width that was set
15972      * @param {Number} adjHeight The box-adjusted height that was set
15973      * @param {Number} rawWidth The width that was originally specified
15974      * @param {Number} rawHeight The height that was originally specified
15975      */
15976     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15977
15978     },
15979
15980     /**
15981      * Called after the component is moved, this method is empty by default but can be implemented by any
15982      * subclass that needs to perform custom logic after a move occurs.
15983      * @param {Number} x The new x position
15984      * @param {Number} y The new y position
15985      */
15986     onPosition : function(x, y){
15987
15988     },
15989
15990     // private
15991     adjustSize : function(w, h){
15992         if(this.autoWidth){
15993             w = 'auto';
15994         }
15995         if(this.autoHeight){
15996             h = 'auto';
15997         }
15998         return {width : w, height: h};
15999     },
16000
16001     // private
16002     adjustPosition : function(x, y){
16003         return {x : x, y: y};
16004     }
16005 });/*
16006  * Original code for Roojs - LGPL
16007  * <script type="text/javascript">
16008  */
16009  
16010 /**
16011  * @class Roo.XComponent
16012  * A delayed Element creator...
16013  * Or a way to group chunks of interface together.
16014  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16015  *  used in conjunction with XComponent.build() it will create an instance of each element,
16016  *  then call addxtype() to build the User interface.
16017  * 
16018  * Mypart.xyx = new Roo.XComponent({
16019
16020     parent : 'Mypart.xyz', // empty == document.element.!!
16021     order : '001',
16022     name : 'xxxx'
16023     region : 'xxxx'
16024     disabled : function() {} 
16025      
16026     tree : function() { // return an tree of xtype declared components
16027         var MODULE = this;
16028         return 
16029         {
16030             xtype : 'NestedLayoutPanel',
16031             // technicall
16032         }
16033      ]
16034  *})
16035  *
16036  *
16037  * It can be used to build a big heiracy, with parent etc.
16038  * or you can just use this to render a single compoent to a dom element
16039  * MYPART.render(Roo.Element | String(id) | dom_element )
16040  *
16041  *
16042  * Usage patterns.
16043  *
16044  * Classic Roo
16045  *
16046  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16047  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16048  *
16049  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16050  *
16051  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16052  * - if mulitple topModules exist, the last one is defined as the top module.
16053  *
16054  * Embeded Roo
16055  * 
16056  * When the top level or multiple modules are to embedded into a existing HTML page,
16057  * the parent element can container '#id' of the element where the module will be drawn.
16058  *
16059  * Bootstrap Roo
16060  *
16061  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16062  * it relies more on a include mechanism, where sub modules are included into an outer page.
16063  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16064  * 
16065  * Bootstrap Roo Included elements
16066  *
16067  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16068  * hence confusing the component builder as it thinks there are multiple top level elements. 
16069  *
16070  * String Over-ride & Translations
16071  *
16072  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16073  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16074  * are needed. @see Roo.XComponent.overlayString  
16075  * 
16076  * 
16077  * 
16078  * @extends Roo.util.Observable
16079  * @constructor
16080  * @param cfg {Object} configuration of component
16081  * 
16082  */
16083 Roo.XComponent = function(cfg) {
16084     Roo.apply(this, cfg);
16085     this.addEvents({ 
16086         /**
16087              * @event built
16088              * Fires when this the componnt is built
16089              * @param {Roo.XComponent} c the component
16090              */
16091         'built' : true
16092         
16093     });
16094     this.region = this.region || 'center'; // default..
16095     Roo.XComponent.register(this);
16096     this.modules = false;
16097     this.el = false; // where the layout goes..
16098     
16099     
16100 }
16101 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16102     /**
16103      * @property el
16104      * The created element (with Roo.factory())
16105      * @type {Roo.Layout}
16106      */
16107     el  : false,
16108     
16109     /**
16110      * @property el
16111      * for BC  - use el in new code
16112      * @type {Roo.Layout}
16113      */
16114     panel : false,
16115     
16116     /**
16117      * @property layout
16118      * for BC  - use el in new code
16119      * @type {Roo.Layout}
16120      */
16121     layout : false,
16122     
16123      /**
16124      * @cfg {Function|boolean} disabled
16125      * If this module is disabled by some rule, return true from the funtion
16126      */
16127     disabled : false,
16128     
16129     /**
16130      * @cfg {String} parent 
16131      * Name of parent element which it get xtype added to..
16132      */
16133     parent: false,
16134     
16135     /**
16136      * @cfg {String} order
16137      * Used to set the order in which elements are created (usefull for multiple tabs)
16138      */
16139     
16140     order : false,
16141     /**
16142      * @cfg {String} name
16143      * String to display while loading.
16144      */
16145     name : false,
16146     /**
16147      * @cfg {String} region
16148      * Region to render component to (defaults to center)
16149      */
16150     region : 'center',
16151     
16152     /**
16153      * @cfg {Array} items
16154      * A single item array - the first element is the root of the tree..
16155      * It's done this way to stay compatible with the Xtype system...
16156      */
16157     items : false,
16158     
16159     /**
16160      * @property _tree
16161      * The method that retuns the tree of parts that make up this compoennt 
16162      * @type {function}
16163      */
16164     _tree  : false,
16165     
16166      /**
16167      * render
16168      * render element to dom or tree
16169      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16170      */
16171     
16172     render : function(el)
16173     {
16174         
16175         el = el || false;
16176         var hp = this.parent ? 1 : 0;
16177         Roo.debug &&  Roo.log(this);
16178         
16179         var tree = this._tree ? this._tree() : this.tree();
16180
16181         
16182         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16183             // if parent is a '#.....' string, then let's use that..
16184             var ename = this.parent.substr(1);
16185             this.parent = false;
16186             Roo.debug && Roo.log(ename);
16187             switch (ename) {
16188                 case 'bootstrap-body':
16189                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16190                         // this is the BorderLayout standard?
16191                        this.parent = { el : true };
16192                        break;
16193                     }
16194                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16195                         // need to insert stuff...
16196                         this.parent =  {
16197                              el : new Roo.bootstrap.layout.Border({
16198                                  el : document.body, 
16199                      
16200                                  center: {
16201                                     titlebar: false,
16202                                     autoScroll:false,
16203                                     closeOnTab: true,
16204                                     tabPosition: 'top',
16205                                       //resizeTabs: true,
16206                                     alwaysShowTabs: true,
16207                                     hideTabs: false
16208                                      //minTabWidth: 140
16209                                  }
16210                              })
16211                         
16212                          };
16213                          break;
16214                     }
16215                          
16216                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16217                         this.parent = { el :  new  Roo.bootstrap.Body() };
16218                         Roo.debug && Roo.log("setting el to doc body");
16219                          
16220                     } else {
16221                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16222                     }
16223                     break;
16224                 case 'bootstrap':
16225                     this.parent = { el : true};
16226                     // fall through
16227                 default:
16228                     el = Roo.get(ename);
16229                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16230                         this.parent = { el : true};
16231                     }
16232                     
16233                     break;
16234             }
16235                 
16236             
16237             if (!el && !this.parent) {
16238                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16239                 return;
16240             }
16241         }
16242         
16243         Roo.debug && Roo.log("EL:");
16244         Roo.debug && Roo.log(el);
16245         Roo.debug && Roo.log("this.parent.el:");
16246         Roo.debug && Roo.log(this.parent.el);
16247         
16248
16249         // altertive root elements ??? - we need a better way to indicate these.
16250         var is_alt = Roo.XComponent.is_alt ||
16251                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16252                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16253                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16254         
16255         
16256         
16257         if (!this.parent && is_alt) {
16258             //el = Roo.get(document.body);
16259             this.parent = { el : true };
16260         }
16261             
16262             
16263         
16264         if (!this.parent) {
16265             
16266             Roo.debug && Roo.log("no parent - creating one");
16267             
16268             el = el ? Roo.get(el) : false;      
16269             
16270             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16271                 
16272                 this.parent =  {
16273                     el : new Roo.bootstrap.layout.Border({
16274                         el: el || document.body,
16275                     
16276                         center: {
16277                             titlebar: false,
16278                             autoScroll:false,
16279                             closeOnTab: true,
16280                             tabPosition: 'top',
16281                              //resizeTabs: true,
16282                             alwaysShowTabs: false,
16283                             hideTabs: true,
16284                             minTabWidth: 140,
16285                             overflow: 'visible'
16286                          }
16287                      })
16288                 };
16289             } else {
16290             
16291                 // it's a top level one..
16292                 this.parent =  {
16293                     el : new Roo.BorderLayout(el || document.body, {
16294                         center: {
16295                             titlebar: false,
16296                             autoScroll:false,
16297                             closeOnTab: true,
16298                             tabPosition: 'top',
16299                              //resizeTabs: true,
16300                             alwaysShowTabs: el && hp? false :  true,
16301                             hideTabs: el || !hp ? true :  false,
16302                             minTabWidth: 140
16303                          }
16304                     })
16305                 };
16306             }
16307         }
16308         
16309         if (!this.parent.el) {
16310                 // probably an old style ctor, which has been disabled.
16311                 return;
16312
16313         }
16314                 // The 'tree' method is  '_tree now' 
16315             
16316         tree.region = tree.region || this.region;
16317         var is_body = false;
16318         if (this.parent.el === true) {
16319             // bootstrap... - body..
16320             if (el) {
16321                 tree.el = el;
16322             }
16323             this.parent.el = Roo.factory(tree);
16324             is_body = true;
16325         }
16326         
16327         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16328         this.fireEvent('built', this);
16329         
16330         this.panel = this.el;
16331         this.layout = this.panel.layout;
16332         this.parentLayout = this.parent.layout  || false;  
16333          
16334     }
16335     
16336 });
16337
16338 Roo.apply(Roo.XComponent, {
16339     /**
16340      * @property  hideProgress
16341      * true to disable the building progress bar.. usefull on single page renders.
16342      * @type Boolean
16343      */
16344     hideProgress : false,
16345     /**
16346      * @property  buildCompleted
16347      * True when the builder has completed building the interface.
16348      * @type Boolean
16349      */
16350     buildCompleted : false,
16351      
16352     /**
16353      * @property  topModule
16354      * the upper most module - uses document.element as it's constructor.
16355      * @type Object
16356      */
16357      
16358     topModule  : false,
16359       
16360     /**
16361      * @property  modules
16362      * array of modules to be created by registration system.
16363      * @type {Array} of Roo.XComponent
16364      */
16365     
16366     modules : [],
16367     /**
16368      * @property  elmodules
16369      * array of modules to be created by which use #ID 
16370      * @type {Array} of Roo.XComponent
16371      */
16372      
16373     elmodules : [],
16374
16375      /**
16376      * @property  is_alt
16377      * Is an alternative Root - normally used by bootstrap or other systems,
16378      *    where the top element in the tree can wrap 'body' 
16379      * @type {boolean}  (default false)
16380      */
16381      
16382     is_alt : false,
16383     /**
16384      * @property  build_from_html
16385      * Build elements from html - used by bootstrap HTML stuff 
16386      *    - this is cleared after build is completed
16387      * @type {boolean}    (default false)
16388      */
16389      
16390     build_from_html : false,
16391     /**
16392      * Register components to be built later.
16393      *
16394      * This solves the following issues
16395      * - Building is not done on page load, but after an authentication process has occured.
16396      * - Interface elements are registered on page load
16397      * - Parent Interface elements may not be loaded before child, so this handles that..
16398      * 
16399      *
16400      * example:
16401      * 
16402      * MyApp.register({
16403           order : '000001',
16404           module : 'Pman.Tab.projectMgr',
16405           region : 'center',
16406           parent : 'Pman.layout',
16407           disabled : false,  // or use a function..
16408         })
16409      
16410      * * @param {Object} details about module
16411      */
16412     register : function(obj) {
16413                 
16414         Roo.XComponent.event.fireEvent('register', obj);
16415         switch(typeof(obj.disabled) ) {
16416                 
16417             case 'undefined':
16418                 break;
16419             
16420             case 'function':
16421                 if ( obj.disabled() ) {
16422                         return;
16423                 }
16424                 break;
16425             
16426             default:
16427                 if (obj.disabled || obj.region == '#disabled') {
16428                         return;
16429                 }
16430                 break;
16431         }
16432                 
16433         this.modules.push(obj);
16434          
16435     },
16436     /**
16437      * convert a string to an object..
16438      * eg. 'AAA.BBB' -> finds AAA.BBB
16439
16440      */
16441     
16442     toObject : function(str)
16443     {
16444         if (!str || typeof(str) == 'object') {
16445             return str;
16446         }
16447         if (str.substring(0,1) == '#') {
16448             return str;
16449         }
16450
16451         var ar = str.split('.');
16452         var rt, o;
16453         rt = ar.shift();
16454             /** eval:var:o */
16455         try {
16456             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16457         } catch (e) {
16458             throw "Module not found : " + str;
16459         }
16460         
16461         if (o === false) {
16462             throw "Module not found : " + str;
16463         }
16464         Roo.each(ar, function(e) {
16465             if (typeof(o[e]) == 'undefined') {
16466                 throw "Module not found : " + str;
16467             }
16468             o = o[e];
16469         });
16470         
16471         return o;
16472         
16473     },
16474     
16475     
16476     /**
16477      * move modules into their correct place in the tree..
16478      * 
16479      */
16480     preBuild : function ()
16481     {
16482         var _t = this;
16483         Roo.each(this.modules , function (obj)
16484         {
16485             Roo.XComponent.event.fireEvent('beforebuild', obj);
16486             
16487             var opar = obj.parent;
16488             try { 
16489                 obj.parent = this.toObject(opar);
16490             } catch(e) {
16491                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16492                 return;
16493             }
16494             
16495             if (!obj.parent) {
16496                 Roo.debug && Roo.log("GOT top level module");
16497                 Roo.debug && Roo.log(obj);
16498                 obj.modules = new Roo.util.MixedCollection(false, 
16499                     function(o) { return o.order + '' }
16500                 );
16501                 this.topModule = obj;
16502                 return;
16503             }
16504                         // parent is a string (usually a dom element name..)
16505             if (typeof(obj.parent) == 'string') {
16506                 this.elmodules.push(obj);
16507                 return;
16508             }
16509             if (obj.parent.constructor != Roo.XComponent) {
16510                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16511             }
16512             if (!obj.parent.modules) {
16513                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16514                     function(o) { return o.order + '' }
16515                 );
16516             }
16517             if (obj.parent.disabled) {
16518                 obj.disabled = true;
16519             }
16520             obj.parent.modules.add(obj);
16521         }, this);
16522     },
16523     
16524      /**
16525      * make a list of modules to build.
16526      * @return {Array} list of modules. 
16527      */ 
16528     
16529     buildOrder : function()
16530     {
16531         var _this = this;
16532         var cmp = function(a,b) {   
16533             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16534         };
16535         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16536             throw "No top level modules to build";
16537         }
16538         
16539         // make a flat list in order of modules to build.
16540         var mods = this.topModule ? [ this.topModule ] : [];
16541                 
16542         
16543         // elmodules (is a list of DOM based modules )
16544         Roo.each(this.elmodules, function(e) {
16545             mods.push(e);
16546             if (!this.topModule &&
16547                 typeof(e.parent) == 'string' &&
16548                 e.parent.substring(0,1) == '#' &&
16549                 Roo.get(e.parent.substr(1))
16550                ) {
16551                 
16552                 _this.topModule = e;
16553             }
16554             
16555         });
16556
16557         
16558         // add modules to their parents..
16559         var addMod = function(m) {
16560             Roo.debug && Roo.log("build Order: add: " + m.name);
16561                 
16562             mods.push(m);
16563             if (m.modules && !m.disabled) {
16564                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16565                 m.modules.keySort('ASC',  cmp );
16566                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16567     
16568                 m.modules.each(addMod);
16569             } else {
16570                 Roo.debug && Roo.log("build Order: no child modules");
16571             }
16572             // not sure if this is used any more..
16573             if (m.finalize) {
16574                 m.finalize.name = m.name + " (clean up) ";
16575                 mods.push(m.finalize);
16576             }
16577             
16578         }
16579         if (this.topModule && this.topModule.modules) { 
16580             this.topModule.modules.keySort('ASC',  cmp );
16581             this.topModule.modules.each(addMod);
16582         } 
16583         return mods;
16584     },
16585     
16586      /**
16587      * Build the registered modules.
16588      * @param {Object} parent element.
16589      * @param {Function} optional method to call after module has been added.
16590      * 
16591      */ 
16592    
16593     build : function(opts) 
16594     {
16595         
16596         if (typeof(opts) != 'undefined') {
16597             Roo.apply(this,opts);
16598         }
16599         
16600         this.preBuild();
16601         var mods = this.buildOrder();
16602       
16603         //this.allmods = mods;
16604         //Roo.debug && Roo.log(mods);
16605         //return;
16606         if (!mods.length) { // should not happen
16607             throw "NO modules!!!";
16608         }
16609         
16610         
16611         var msg = "Building Interface...";
16612         // flash it up as modal - so we store the mask!?
16613         if (!this.hideProgress && Roo.MessageBox) {
16614             Roo.MessageBox.show({ title: 'loading' });
16615             Roo.MessageBox.show({
16616                title: "Please wait...",
16617                msg: msg,
16618                width:450,
16619                progress:true,
16620                buttons : false,
16621                closable:false,
16622                modal: false
16623               
16624             });
16625         }
16626         var total = mods.length;
16627         
16628         var _this = this;
16629         var progressRun = function() {
16630             if (!mods.length) {
16631                 Roo.debug && Roo.log('hide?');
16632                 if (!this.hideProgress && Roo.MessageBox) {
16633                     Roo.MessageBox.hide();
16634                 }
16635                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16636                 
16637                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16638                 
16639                 // THE END...
16640                 return false;   
16641             }
16642             
16643             var m = mods.shift();
16644             
16645             
16646             Roo.debug && Roo.log(m);
16647             // not sure if this is supported any more.. - modules that are are just function
16648             if (typeof(m) == 'function') { 
16649                 m.call(this);
16650                 return progressRun.defer(10, _this);
16651             } 
16652             
16653             
16654             msg = "Building Interface " + (total  - mods.length) + 
16655                     " of " + total + 
16656                     (m.name ? (' - ' + m.name) : '');
16657                         Roo.debug && Roo.log(msg);
16658             if (!_this.hideProgress &&  Roo.MessageBox) { 
16659                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16660             }
16661             
16662          
16663             // is the module disabled?
16664             var disabled = (typeof(m.disabled) == 'function') ?
16665                 m.disabled.call(m.module.disabled) : m.disabled;    
16666             
16667             
16668             if (disabled) {
16669                 return progressRun(); // we do not update the display!
16670             }
16671             
16672             // now build 
16673             
16674                         
16675                         
16676             m.render();
16677             // it's 10 on top level, and 1 on others??? why...
16678             return progressRun.defer(10, _this);
16679              
16680         }
16681         progressRun.defer(1, _this);
16682      
16683         
16684         
16685     },
16686     /**
16687      * Overlay a set of modified strings onto a component
16688      * This is dependant on our builder exporting the strings and 'named strings' elements.
16689      * 
16690      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16691      * @param {Object} associative array of 'named' string and it's new value.
16692      * 
16693      */
16694         overlayStrings : function( component, strings )
16695     {
16696         if (typeof(component['_named_strings']) == 'undefined') {
16697             throw "ERROR: component does not have _named_strings";
16698         }
16699         for ( var k in strings ) {
16700             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16701             if (md !== false) {
16702                 component['_strings'][md] = strings[k];
16703             } else {
16704                 Roo.log('could not find named string: ' + k + ' in');
16705                 Roo.log(component);
16706             }
16707             
16708         }
16709         
16710     },
16711     
16712         
16713         /**
16714          * Event Object.
16715          *
16716          *
16717          */
16718         event: false, 
16719     /**
16720          * wrapper for event.on - aliased later..  
16721          * Typically use to register a event handler for register:
16722          *
16723          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16724          *
16725          */
16726     on : false
16727    
16728     
16729     
16730 });
16731
16732 Roo.XComponent.event = new Roo.util.Observable({
16733                 events : { 
16734                         /**
16735                          * @event register
16736                          * Fires when an Component is registered,
16737                          * set the disable property on the Component to stop registration.
16738                          * @param {Roo.XComponent} c the component being registerd.
16739                          * 
16740                          */
16741                         'register' : true,
16742             /**
16743                          * @event beforebuild
16744                          * Fires before each Component is built
16745                          * can be used to apply permissions.
16746                          * @param {Roo.XComponent} c the component being registerd.
16747                          * 
16748                          */
16749                         'beforebuild' : true,
16750                         /**
16751                          * @event buildcomplete
16752                          * Fires on the top level element when all elements have been built
16753                          * @param {Roo.XComponent} the top level component.
16754                          */
16755                         'buildcomplete' : true
16756                         
16757                 }
16758 });
16759
16760 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16761  //
16762  /**
16763  * marked - a markdown parser
16764  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16765  * https://github.com/chjj/marked
16766  */
16767
16768
16769 /**
16770  *
16771  * Roo.Markdown - is a very crude wrapper around marked..
16772  *
16773  * usage:
16774  * 
16775  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16776  * 
16777  * Note: move the sample code to the bottom of this
16778  * file before uncommenting it.
16779  *
16780  */
16781
16782 Roo.Markdown = {};
16783 Roo.Markdown.toHtml = function(text) {
16784     
16785     var c = new Roo.Markdown.marked.setOptions({
16786             renderer: new Roo.Markdown.marked.Renderer(),
16787             gfm: true,
16788             tables: true,
16789             breaks: false,
16790             pedantic: false,
16791             sanitize: false,
16792             smartLists: true,
16793             smartypants: false
16794           });
16795     // A FEW HACKS!!?
16796     
16797     text = text.replace(/\\\n/g,' ');
16798     return Roo.Markdown.marked(text);
16799 };
16800 //
16801 // converter
16802 //
16803 // Wraps all "globals" so that the only thing
16804 // exposed is makeHtml().
16805 //
16806 (function() {
16807     
16808      /**
16809          * eval:var:escape
16810          * eval:var:unescape
16811          * eval:var:replace
16812          */
16813       
16814     /**
16815      * Helpers
16816      */
16817     
16818     var escape = function (html, encode) {
16819       return html
16820         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
16821         .replace(/</g, '&lt;')
16822         .replace(/>/g, '&gt;')
16823         .replace(/"/g, '&quot;')
16824         .replace(/'/g, '&#39;');
16825     }
16826     
16827     var unescape = function (html) {
16828         // explicitly match decimal, hex, and named HTML entities 
16829       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
16830         n = n.toLowerCase();
16831         if (n === 'colon') { return ':'; }
16832         if (n.charAt(0) === '#') {
16833           return n.charAt(1) === 'x'
16834             ? String.fromCharCode(parseInt(n.substring(2), 16))
16835             : String.fromCharCode(+n.substring(1));
16836         }
16837         return '';
16838       });
16839     }
16840     
16841     var replace = function (regex, opt) {
16842       regex = regex.source;
16843       opt = opt || '';
16844       return function self(name, val) {
16845         if (!name) { return new RegExp(regex, opt); }
16846         val = val.source || val;
16847         val = val.replace(/(^|[^\[])\^/g, '$1');
16848         regex = regex.replace(name, val);
16849         return self;
16850       };
16851     }
16852
16853
16854          /**
16855          * eval:var:noop
16856     */
16857     var noop = function () {}
16858     noop.exec = noop;
16859     
16860          /**
16861          * eval:var:merge
16862     */
16863     var merge = function (obj) {
16864       var i = 1
16865         , target
16866         , key;
16867     
16868       for (; i < arguments.length; i++) {
16869         target = arguments[i];
16870         for (key in target) {
16871           if (Object.prototype.hasOwnProperty.call(target, key)) {
16872             obj[key] = target[key];
16873           }
16874         }
16875       }
16876     
16877       return obj;
16878     }
16879     
16880     
16881     /**
16882      * Block-Level Grammar
16883      */
16884     
16885     
16886     
16887     
16888     var block = {
16889       newline: /^\n+/,
16890       code: /^( {4}[^\n]+\n*)+/,
16891       fences: noop,
16892       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16893       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16894       nptable: noop,
16895       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16896       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16897       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16898       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16899       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16900       table: noop,
16901       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16902       text: /^[^\n]+/
16903     };
16904     
16905     block.bullet = /(?:[*+-]|\d+\.)/;
16906     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16907     block.item = replace(block.item, 'gm')
16908       (/bull/g, block.bullet)
16909       ();
16910     
16911     block.list = replace(block.list)
16912       (/bull/g, block.bullet)
16913       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16914       ('def', '\\n+(?=' + block.def.source + ')')
16915       ();
16916     
16917     block.blockquote = replace(block.blockquote)
16918       ('def', block.def)
16919       ();
16920     
16921     block._tag = '(?!(?:'
16922       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16923       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16924       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16925     
16926     block.html = replace(block.html)
16927       ('comment', /<!--[\s\S]*?-->/)
16928       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16929       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16930       (/tag/g, block._tag)
16931       ();
16932     
16933     block.paragraph = replace(block.paragraph)
16934       ('hr', block.hr)
16935       ('heading', block.heading)
16936       ('lheading', block.lheading)
16937       ('blockquote', block.blockquote)
16938       ('tag', '<' + block._tag)
16939       ('def', block.def)
16940       ();
16941     
16942     /**
16943      * Normal Block Grammar
16944      */
16945     
16946     block.normal = merge({}, block);
16947     
16948     /**
16949      * GFM Block Grammar
16950      */
16951     
16952     block.gfm = merge({}, block.normal, {
16953       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16954       paragraph: /^/,
16955       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16956     });
16957     
16958     block.gfm.paragraph = replace(block.paragraph)
16959       ('(?!', '(?!'
16960         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16961         + block.list.source.replace('\\1', '\\3') + '|')
16962       ();
16963     
16964     /**
16965      * GFM + Tables Block Grammar
16966      */
16967     
16968     block.tables = merge({}, block.gfm, {
16969       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16970       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16971     });
16972     
16973     /**
16974      * Block Lexer
16975      */
16976     
16977     var Lexer = function (options) {
16978       this.tokens = [];
16979       this.tokens.links = {};
16980       this.options = options || marked.defaults;
16981       this.rules = block.normal;
16982     
16983       if (this.options.gfm) {
16984         if (this.options.tables) {
16985           this.rules = block.tables;
16986         } else {
16987           this.rules = block.gfm;
16988         }
16989       }
16990     }
16991     
16992     /**
16993      * Expose Block Rules
16994      */
16995     
16996     Lexer.rules = block;
16997     
16998     /**
16999      * Static Lex Method
17000      */
17001     
17002     Lexer.lex = function(src, options) {
17003       var lexer = new Lexer(options);
17004       return lexer.lex(src);
17005     };
17006     
17007     /**
17008      * Preprocessing
17009      */
17010     
17011     Lexer.prototype.lex = function(src) {
17012       src = src
17013         .replace(/\r\n|\r/g, '\n')
17014         .replace(/\t/g, '    ')
17015         .replace(/\u00a0/g, ' ')
17016         .replace(/\u2424/g, '\n');
17017     
17018       return this.token(src, true);
17019     };
17020     
17021     /**
17022      * Lexing
17023      */
17024     
17025     Lexer.prototype.token = function(src, top, bq) {
17026       var src = src.replace(/^ +$/gm, '')
17027         , next
17028         , loose
17029         , cap
17030         , bull
17031         , b
17032         , item
17033         , space
17034         , i
17035         , l;
17036     
17037       while (src) {
17038         // newline
17039         if (cap = this.rules.newline.exec(src)) {
17040           src = src.substring(cap[0].length);
17041           if (cap[0].length > 1) {
17042             this.tokens.push({
17043               type: 'space'
17044             });
17045           }
17046         }
17047     
17048         // code
17049         if (cap = this.rules.code.exec(src)) {
17050           src = src.substring(cap[0].length);
17051           cap = cap[0].replace(/^ {4}/gm, '');
17052           this.tokens.push({
17053             type: 'code',
17054             text: !this.options.pedantic
17055               ? cap.replace(/\n+$/, '')
17056               : cap
17057           });
17058           continue;
17059         }
17060     
17061         // fences (gfm)
17062         if (cap = this.rules.fences.exec(src)) {
17063           src = src.substring(cap[0].length);
17064           this.tokens.push({
17065             type: 'code',
17066             lang: cap[2],
17067             text: cap[3] || ''
17068           });
17069           continue;
17070         }
17071     
17072         // heading
17073         if (cap = this.rules.heading.exec(src)) {
17074           src = src.substring(cap[0].length);
17075           this.tokens.push({
17076             type: 'heading',
17077             depth: cap[1].length,
17078             text: cap[2]
17079           });
17080           continue;
17081         }
17082     
17083         // table no leading pipe (gfm)
17084         if (top && (cap = this.rules.nptable.exec(src))) {
17085           src = src.substring(cap[0].length);
17086     
17087           item = {
17088             type: 'table',
17089             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17090             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17091             cells: cap[3].replace(/\n$/, '').split('\n')
17092           };
17093     
17094           for (i = 0; i < item.align.length; i++) {
17095             if (/^ *-+: *$/.test(item.align[i])) {
17096               item.align[i] = 'right';
17097             } else if (/^ *:-+: *$/.test(item.align[i])) {
17098               item.align[i] = 'center';
17099             } else if (/^ *:-+ *$/.test(item.align[i])) {
17100               item.align[i] = 'left';
17101             } else {
17102               item.align[i] = null;
17103             }
17104           }
17105     
17106           for (i = 0; i < item.cells.length; i++) {
17107             item.cells[i] = item.cells[i].split(/ *\| */);
17108           }
17109     
17110           this.tokens.push(item);
17111     
17112           continue;
17113         }
17114     
17115         // lheading
17116         if (cap = this.rules.lheading.exec(src)) {
17117           src = src.substring(cap[0].length);
17118           this.tokens.push({
17119             type: 'heading',
17120             depth: cap[2] === '=' ? 1 : 2,
17121             text: cap[1]
17122           });
17123           continue;
17124         }
17125     
17126         // hr
17127         if (cap = this.rules.hr.exec(src)) {
17128           src = src.substring(cap[0].length);
17129           this.tokens.push({
17130             type: 'hr'
17131           });
17132           continue;
17133         }
17134     
17135         // blockquote
17136         if (cap = this.rules.blockquote.exec(src)) {
17137           src = src.substring(cap[0].length);
17138     
17139           this.tokens.push({
17140             type: 'blockquote_start'
17141           });
17142     
17143           cap = cap[0].replace(/^ *> ?/gm, '');
17144     
17145           // Pass `top` to keep the current
17146           // "toplevel" state. This is exactly
17147           // how markdown.pl works.
17148           this.token(cap, top, true);
17149     
17150           this.tokens.push({
17151             type: 'blockquote_end'
17152           });
17153     
17154           continue;
17155         }
17156     
17157         // list
17158         if (cap = this.rules.list.exec(src)) {
17159           src = src.substring(cap[0].length);
17160           bull = cap[2];
17161     
17162           this.tokens.push({
17163             type: 'list_start',
17164             ordered: bull.length > 1
17165           });
17166     
17167           // Get each top-level item.
17168           cap = cap[0].match(this.rules.item);
17169     
17170           next = false;
17171           l = cap.length;
17172           i = 0;
17173     
17174           for (; i < l; i++) {
17175             item = cap[i];
17176     
17177             // Remove the list item's bullet
17178             // so it is seen as the next token.
17179             space = item.length;
17180             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17181     
17182             // Outdent whatever the
17183             // list item contains. Hacky.
17184             if (~item.indexOf('\n ')) {
17185               space -= item.length;
17186               item = !this.options.pedantic
17187                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17188                 : item.replace(/^ {1,4}/gm, '');
17189             }
17190     
17191             // Determine whether the next list item belongs here.
17192             // Backpedal if it does not belong in this list.
17193             if (this.options.smartLists && i !== l - 1) {
17194               b = block.bullet.exec(cap[i + 1])[0];
17195               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17196                 src = cap.slice(i + 1).join('\n') + src;
17197                 i = l - 1;
17198               }
17199             }
17200     
17201             // Determine whether item is loose or not.
17202             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17203             // for discount behavior.
17204             loose = next || /\n\n(?!\s*$)/.test(item);
17205             if (i !== l - 1) {
17206               next = item.charAt(item.length - 1) === '\n';
17207               if (!loose) { loose = next; }
17208             }
17209     
17210             this.tokens.push({
17211               type: loose
17212                 ? 'loose_item_start'
17213                 : 'list_item_start'
17214             });
17215     
17216             // Recurse.
17217             this.token(item, false, bq);
17218     
17219             this.tokens.push({
17220               type: 'list_item_end'
17221             });
17222           }
17223     
17224           this.tokens.push({
17225             type: 'list_end'
17226           });
17227     
17228           continue;
17229         }
17230     
17231         // html
17232         if (cap = this.rules.html.exec(src)) {
17233           src = src.substring(cap[0].length);
17234           this.tokens.push({
17235             type: this.options.sanitize
17236               ? 'paragraph'
17237               : 'html',
17238             pre: !this.options.sanitizer
17239               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17240             text: cap[0]
17241           });
17242           continue;
17243         }
17244     
17245         // def
17246         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17247           src = src.substring(cap[0].length);
17248           this.tokens.links[cap[1].toLowerCase()] = {
17249             href: cap[2],
17250             title: cap[3]
17251           };
17252           continue;
17253         }
17254     
17255         // table (gfm)
17256         if (top && (cap = this.rules.table.exec(src))) {
17257           src = src.substring(cap[0].length);
17258     
17259           item = {
17260             type: 'table',
17261             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17262             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17263             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17264           };
17265     
17266           for (i = 0; i < item.align.length; i++) {
17267             if (/^ *-+: *$/.test(item.align[i])) {
17268               item.align[i] = 'right';
17269             } else if (/^ *:-+: *$/.test(item.align[i])) {
17270               item.align[i] = 'center';
17271             } else if (/^ *:-+ *$/.test(item.align[i])) {
17272               item.align[i] = 'left';
17273             } else {
17274               item.align[i] = null;
17275             }
17276           }
17277     
17278           for (i = 0; i < item.cells.length; i++) {
17279             item.cells[i] = item.cells[i]
17280               .replace(/^ *\| *| *\| *$/g, '')
17281               .split(/ *\| */);
17282           }
17283     
17284           this.tokens.push(item);
17285     
17286           continue;
17287         }
17288     
17289         // top-level paragraph
17290         if (top && (cap = this.rules.paragraph.exec(src))) {
17291           src = src.substring(cap[0].length);
17292           this.tokens.push({
17293             type: 'paragraph',
17294             text: cap[1].charAt(cap[1].length - 1) === '\n'
17295               ? cap[1].slice(0, -1)
17296               : cap[1]
17297           });
17298           continue;
17299         }
17300     
17301         // text
17302         if (cap = this.rules.text.exec(src)) {
17303           // Top-level should never reach here.
17304           src = src.substring(cap[0].length);
17305           this.tokens.push({
17306             type: 'text',
17307             text: cap[0]
17308           });
17309           continue;
17310         }
17311     
17312         if (src) {
17313           throw new
17314             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17315         }
17316       }
17317     
17318       return this.tokens;
17319     };
17320     
17321     /**
17322      * Inline-Level Grammar
17323      */
17324     
17325     var inline = {
17326       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17327       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17328       url: noop,
17329       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17330       link: /^!?\[(inside)\]\(href\)/,
17331       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17332       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17333       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17334       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17335       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17336       br: /^ {2,}\n(?!\s*$)/,
17337       del: noop,
17338       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17339     };
17340     
17341     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17342     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17343     
17344     inline.link = replace(inline.link)
17345       ('inside', inline._inside)
17346       ('href', inline._href)
17347       ();
17348     
17349     inline.reflink = replace(inline.reflink)
17350       ('inside', inline._inside)
17351       ();
17352     
17353     /**
17354      * Normal Inline Grammar
17355      */
17356     
17357     inline.normal = merge({}, inline);
17358     
17359     /**
17360      * Pedantic Inline Grammar
17361      */
17362     
17363     inline.pedantic = merge({}, inline.normal, {
17364       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17365       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17366     });
17367     
17368     /**
17369      * GFM Inline Grammar
17370      */
17371     
17372     inline.gfm = merge({}, inline.normal, {
17373       escape: replace(inline.escape)('])', '~|])')(),
17374       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17375       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17376       text: replace(inline.text)
17377         (']|', '~]|')
17378         ('|', '|https?://|')
17379         ()
17380     });
17381     
17382     /**
17383      * GFM + Line Breaks Inline Grammar
17384      */
17385     
17386     inline.breaks = merge({}, inline.gfm, {
17387       br: replace(inline.br)('{2,}', '*')(),
17388       text: replace(inline.gfm.text)('{2,}', '*')()
17389     });
17390     
17391     /**
17392      * Inline Lexer & Compiler
17393      */
17394     
17395     var InlineLexer  = function (links, options) {
17396       this.options = options || marked.defaults;
17397       this.links = links;
17398       this.rules = inline.normal;
17399       this.renderer = this.options.renderer || new Renderer;
17400       this.renderer.options = this.options;
17401     
17402       if (!this.links) {
17403         throw new
17404           Error('Tokens array requires a `links` property.');
17405       }
17406     
17407       if (this.options.gfm) {
17408         if (this.options.breaks) {
17409           this.rules = inline.breaks;
17410         } else {
17411           this.rules = inline.gfm;
17412         }
17413       } else if (this.options.pedantic) {
17414         this.rules = inline.pedantic;
17415       }
17416     }
17417     
17418     /**
17419      * Expose Inline Rules
17420      */
17421     
17422     InlineLexer.rules = inline;
17423     
17424     /**
17425      * Static Lexing/Compiling Method
17426      */
17427     
17428     InlineLexer.output = function(src, links, options) {
17429       var inline = new InlineLexer(links, options);
17430       return inline.output(src);
17431     };
17432     
17433     /**
17434      * Lexing/Compiling
17435      */
17436     
17437     InlineLexer.prototype.output = function(src) {
17438       var out = ''
17439         , link
17440         , text
17441         , href
17442         , cap;
17443     
17444       while (src) {
17445         // escape
17446         if (cap = this.rules.escape.exec(src)) {
17447           src = src.substring(cap[0].length);
17448           out += cap[1];
17449           continue;
17450         }
17451     
17452         // autolink
17453         if (cap = this.rules.autolink.exec(src)) {
17454           src = src.substring(cap[0].length);
17455           if (cap[2] === '@') {
17456             text = cap[1].charAt(6) === ':'
17457               ? this.mangle(cap[1].substring(7))
17458               : this.mangle(cap[1]);
17459             href = this.mangle('mailto:') + text;
17460           } else {
17461             text = escape(cap[1]);
17462             href = text;
17463           }
17464           out += this.renderer.link(href, null, text);
17465           continue;
17466         }
17467     
17468         // url (gfm)
17469         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17470           src = src.substring(cap[0].length);
17471           text = escape(cap[1]);
17472           href = text;
17473           out += this.renderer.link(href, null, text);
17474           continue;
17475         }
17476     
17477         // tag
17478         if (cap = this.rules.tag.exec(src)) {
17479           if (!this.inLink && /^<a /i.test(cap[0])) {
17480             this.inLink = true;
17481           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17482             this.inLink = false;
17483           }
17484           src = src.substring(cap[0].length);
17485           out += this.options.sanitize
17486             ? this.options.sanitizer
17487               ? this.options.sanitizer(cap[0])
17488               : escape(cap[0])
17489             : cap[0];
17490           continue;
17491         }
17492     
17493         // link
17494         if (cap = this.rules.link.exec(src)) {
17495           src = src.substring(cap[0].length);
17496           this.inLink = true;
17497           out += this.outputLink(cap, {
17498             href: cap[2],
17499             title: cap[3]
17500           });
17501           this.inLink = false;
17502           continue;
17503         }
17504     
17505         // reflink, nolink
17506         if ((cap = this.rules.reflink.exec(src))
17507             || (cap = this.rules.nolink.exec(src))) {
17508           src = src.substring(cap[0].length);
17509           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17510           link = this.links[link.toLowerCase()];
17511           if (!link || !link.href) {
17512             out += cap[0].charAt(0);
17513             src = cap[0].substring(1) + src;
17514             continue;
17515           }
17516           this.inLink = true;
17517           out += this.outputLink(cap, link);
17518           this.inLink = false;
17519           continue;
17520         }
17521     
17522         // strong
17523         if (cap = this.rules.strong.exec(src)) {
17524           src = src.substring(cap[0].length);
17525           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17526           continue;
17527         }
17528     
17529         // em
17530         if (cap = this.rules.em.exec(src)) {
17531           src = src.substring(cap[0].length);
17532           out += this.renderer.em(this.output(cap[2] || cap[1]));
17533           continue;
17534         }
17535     
17536         // code
17537         if (cap = this.rules.code.exec(src)) {
17538           src = src.substring(cap[0].length);
17539           out += this.renderer.codespan(escape(cap[2], true));
17540           continue;
17541         }
17542     
17543         // br
17544         if (cap = this.rules.br.exec(src)) {
17545           src = src.substring(cap[0].length);
17546           out += this.renderer.br();
17547           continue;
17548         }
17549     
17550         // del (gfm)
17551         if (cap = this.rules.del.exec(src)) {
17552           src = src.substring(cap[0].length);
17553           out += this.renderer.del(this.output(cap[1]));
17554           continue;
17555         }
17556     
17557         // text
17558         if (cap = this.rules.text.exec(src)) {
17559           src = src.substring(cap[0].length);
17560           out += this.renderer.text(escape(this.smartypants(cap[0])));
17561           continue;
17562         }
17563     
17564         if (src) {
17565           throw new
17566             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17567         }
17568       }
17569     
17570       return out;
17571     };
17572     
17573     /**
17574      * Compile Link
17575      */
17576     
17577     InlineLexer.prototype.outputLink = function(cap, link) {
17578       var href = escape(link.href)
17579         , title = link.title ? escape(link.title) : null;
17580     
17581       return cap[0].charAt(0) !== '!'
17582         ? this.renderer.link(href, title, this.output(cap[1]))
17583         : this.renderer.image(href, title, escape(cap[1]));
17584     };
17585     
17586     /**
17587      * Smartypants Transformations
17588      */
17589     
17590     InlineLexer.prototype.smartypants = function(text) {
17591       if (!this.options.smartypants)  { return text; }
17592       return text
17593         // em-dashes
17594         .replace(/---/g, '\u2014')
17595         // en-dashes
17596         .replace(/--/g, '\u2013')
17597         // opening singles
17598         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17599         // closing singles & apostrophes
17600         .replace(/'/g, '\u2019')
17601         // opening doubles
17602         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17603         // closing doubles
17604         .replace(/"/g, '\u201d')
17605         // ellipses
17606         .replace(/\.{3}/g, '\u2026');
17607     };
17608     
17609     /**
17610      * Mangle Links
17611      */
17612     
17613     InlineLexer.prototype.mangle = function(text) {
17614       if (!this.options.mangle) { return text; }
17615       var out = ''
17616         , l = text.length
17617         , i = 0
17618         , ch;
17619     
17620       for (; i < l; i++) {
17621         ch = text.charCodeAt(i);
17622         if (Math.random() > 0.5) {
17623           ch = 'x' + ch.toString(16);
17624         }
17625         out += '&#' + ch + ';';
17626       }
17627     
17628       return out;
17629     };
17630     
17631     /**
17632      * Renderer
17633      */
17634     
17635      /**
17636          * eval:var:Renderer
17637     */
17638     
17639     var Renderer   = function (options) {
17640       this.options = options || {};
17641     }
17642     
17643     Renderer.prototype.code = function(code, lang, escaped) {
17644       if (this.options.highlight) {
17645         var out = this.options.highlight(code, lang);
17646         if (out != null && out !== code) {
17647           escaped = true;
17648           code = out;
17649         }
17650       } else {
17651             // hack!!! - it's already escapeD?
17652             escaped = true;
17653       }
17654     
17655       if (!lang) {
17656         return '<pre><code>'
17657           + (escaped ? code : escape(code, true))
17658           + '\n</code></pre>';
17659       }
17660     
17661       return '<pre><code class="'
17662         + this.options.langPrefix
17663         + escape(lang, true)
17664         + '">'
17665         + (escaped ? code : escape(code, true))
17666         + '\n</code></pre>\n';
17667     };
17668     
17669     Renderer.prototype.blockquote = function(quote) {
17670       return '<blockquote>\n' + quote + '</blockquote>\n';
17671     };
17672     
17673     Renderer.prototype.html = function(html) {
17674       return html;
17675     };
17676     
17677     Renderer.prototype.heading = function(text, level, raw) {
17678       return '<h'
17679         + level
17680         + ' id="'
17681         + this.options.headerPrefix
17682         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17683         + '">'
17684         + text
17685         + '</h'
17686         + level
17687         + '>\n';
17688     };
17689     
17690     Renderer.prototype.hr = function() {
17691       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17692     };
17693     
17694     Renderer.prototype.list = function(body, ordered) {
17695       var type = ordered ? 'ol' : 'ul';
17696       return '<' + type + '>\n' + body + '</' + type + '>\n';
17697     };
17698     
17699     Renderer.prototype.listitem = function(text) {
17700       return '<li>' + text + '</li>\n';
17701     };
17702     
17703     Renderer.prototype.paragraph = function(text) {
17704       return '<p>' + text + '</p>\n';
17705     };
17706     
17707     Renderer.prototype.table = function(header, body) {
17708       return '<table class="table table-striped">\n'
17709         + '<thead>\n'
17710         + header
17711         + '</thead>\n'
17712         + '<tbody>\n'
17713         + body
17714         + '</tbody>\n'
17715         + '</table>\n';
17716     };
17717     
17718     Renderer.prototype.tablerow = function(content) {
17719       return '<tr>\n' + content + '</tr>\n';
17720     };
17721     
17722     Renderer.prototype.tablecell = function(content, flags) {
17723       var type = flags.header ? 'th' : 'td';
17724       var tag = flags.align
17725         ? '<' + type + ' style="text-align:' + flags.align + '">'
17726         : '<' + type + '>';
17727       return tag + content + '</' + type + '>\n';
17728     };
17729     
17730     // span level renderer
17731     Renderer.prototype.strong = function(text) {
17732       return '<strong>' + text + '</strong>';
17733     };
17734     
17735     Renderer.prototype.em = function(text) {
17736       return '<em>' + text + '</em>';
17737     };
17738     
17739     Renderer.prototype.codespan = function(text) {
17740       return '<code>' + text + '</code>';
17741     };
17742     
17743     Renderer.prototype.br = function() {
17744       return this.options.xhtml ? '<br/>' : '<br>';
17745     };
17746     
17747     Renderer.prototype.del = function(text) {
17748       return '<del>' + text + '</del>';
17749     };
17750     
17751     Renderer.prototype.link = function(href, title, text) {
17752       if (this.options.sanitize) {
17753         try {
17754           var prot = decodeURIComponent(unescape(href))
17755             .replace(/[^\w:]/g, '')
17756             .toLowerCase();
17757         } catch (e) {
17758           return '';
17759         }
17760         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17761           return '';
17762         }
17763       }
17764       var out = '<a href="' + href + '"';
17765       if (title) {
17766         out += ' title="' + title + '"';
17767       }
17768       out += '>' + text + '</a>';
17769       return out;
17770     };
17771     
17772     Renderer.prototype.image = function(href, title, text) {
17773       var out = '<img src="' + href + '" alt="' + text + '"';
17774       if (title) {
17775         out += ' title="' + title + '"';
17776       }
17777       out += this.options.xhtml ? '/>' : '>';
17778       return out;
17779     };
17780     
17781     Renderer.prototype.text = function(text) {
17782       return text;
17783     };
17784     
17785     /**
17786      * Parsing & Compiling
17787      */
17788          /**
17789          * eval:var:Parser
17790     */
17791     
17792     var Parser= function (options) {
17793       this.tokens = [];
17794       this.token = null;
17795       this.options = options || marked.defaults;
17796       this.options.renderer = this.options.renderer || new Renderer;
17797       this.renderer = this.options.renderer;
17798       this.renderer.options = this.options;
17799     }
17800     
17801     /**
17802      * Static Parse Method
17803      */
17804     
17805     Parser.parse = function(src, options, renderer) {
17806       var parser = new Parser(options, renderer);
17807       return parser.parse(src);
17808     };
17809     
17810     /**
17811      * Parse Loop
17812      */
17813     
17814     Parser.prototype.parse = function(src) {
17815       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17816       this.tokens = src.reverse();
17817     
17818       var out = '';
17819       while (this.next()) {
17820         out += this.tok();
17821       }
17822     
17823       return out;
17824     };
17825     
17826     /**
17827      * Next Token
17828      */
17829     
17830     Parser.prototype.next = function() {
17831       return this.token = this.tokens.pop();
17832     };
17833     
17834     /**
17835      * Preview Next Token
17836      */
17837     
17838     Parser.prototype.peek = function() {
17839       return this.tokens[this.tokens.length - 1] || 0;
17840     };
17841     
17842     /**
17843      * Parse Text Tokens
17844      */
17845     
17846     Parser.prototype.parseText = function() {
17847       var body = this.token.text;
17848     
17849       while (this.peek().type === 'text') {
17850         body += '\n' + this.next().text;
17851       }
17852     
17853       return this.inline.output(body);
17854     };
17855     
17856     /**
17857      * Parse Current Token
17858      */
17859     
17860     Parser.prototype.tok = function() {
17861       switch (this.token.type) {
17862         case 'space': {
17863           return '';
17864         }
17865         case 'hr': {
17866           return this.renderer.hr();
17867         }
17868         case 'heading': {
17869           return this.renderer.heading(
17870             this.inline.output(this.token.text),
17871             this.token.depth,
17872             this.token.text);
17873         }
17874         case 'code': {
17875           return this.renderer.code(this.token.text,
17876             this.token.lang,
17877             this.token.escaped);
17878         }
17879         case 'table': {
17880           var header = ''
17881             , body = ''
17882             , i
17883             , row
17884             , cell
17885             , flags
17886             , j;
17887     
17888           // header
17889           cell = '';
17890           for (i = 0; i < this.token.header.length; i++) {
17891             flags = { header: true, align: this.token.align[i] };
17892             cell += this.renderer.tablecell(
17893               this.inline.output(this.token.header[i]),
17894               { header: true, align: this.token.align[i] }
17895             );
17896           }
17897           header += this.renderer.tablerow(cell);
17898     
17899           for (i = 0; i < this.token.cells.length; i++) {
17900             row = this.token.cells[i];
17901     
17902             cell = '';
17903             for (j = 0; j < row.length; j++) {
17904               cell += this.renderer.tablecell(
17905                 this.inline.output(row[j]),
17906                 { header: false, align: this.token.align[j] }
17907               );
17908             }
17909     
17910             body += this.renderer.tablerow(cell);
17911           }
17912           return this.renderer.table(header, body);
17913         }
17914         case 'blockquote_start': {
17915           var body = '';
17916     
17917           while (this.next().type !== 'blockquote_end') {
17918             body += this.tok();
17919           }
17920     
17921           return this.renderer.blockquote(body);
17922         }
17923         case 'list_start': {
17924           var body = ''
17925             , ordered = this.token.ordered;
17926     
17927           while (this.next().type !== 'list_end') {
17928             body += this.tok();
17929           }
17930     
17931           return this.renderer.list(body, ordered);
17932         }
17933         case 'list_item_start': {
17934           var body = '';
17935     
17936           while (this.next().type !== 'list_item_end') {
17937             body += this.token.type === 'text'
17938               ? this.parseText()
17939               : this.tok();
17940           }
17941     
17942           return this.renderer.listitem(body);
17943         }
17944         case 'loose_item_start': {
17945           var body = '';
17946     
17947           while (this.next().type !== 'list_item_end') {
17948             body += this.tok();
17949           }
17950     
17951           return this.renderer.listitem(body);
17952         }
17953         case 'html': {
17954           var html = !this.token.pre && !this.options.pedantic
17955             ? this.inline.output(this.token.text)
17956             : this.token.text;
17957           return this.renderer.html(html);
17958         }
17959         case 'paragraph': {
17960           return this.renderer.paragraph(this.inline.output(this.token.text));
17961         }
17962         case 'text': {
17963           return this.renderer.paragraph(this.parseText());
17964         }
17965       }
17966     };
17967   
17968     
17969     /**
17970      * Marked
17971      */
17972          /**
17973          * eval:var:marked
17974     */
17975     var marked = function (src, opt, callback) {
17976       if (callback || typeof opt === 'function') {
17977         if (!callback) {
17978           callback = opt;
17979           opt = null;
17980         }
17981     
17982         opt = merge({}, marked.defaults, opt || {});
17983     
17984         var highlight = opt.highlight
17985           , tokens
17986           , pending
17987           , i = 0;
17988     
17989         try {
17990           tokens = Lexer.lex(src, opt)
17991         } catch (e) {
17992           return callback(e);
17993         }
17994     
17995         pending = tokens.length;
17996          /**
17997          * eval:var:done
17998     */
17999         var done = function(err) {
18000           if (err) {
18001             opt.highlight = highlight;
18002             return callback(err);
18003           }
18004     
18005           var out;
18006     
18007           try {
18008             out = Parser.parse(tokens, opt);
18009           } catch (e) {
18010             err = e;
18011           }
18012     
18013           opt.highlight = highlight;
18014     
18015           return err
18016             ? callback(err)
18017             : callback(null, out);
18018         };
18019     
18020         if (!highlight || highlight.length < 3) {
18021           return done();
18022         }
18023     
18024         delete opt.highlight;
18025     
18026         if (!pending) { return done(); }
18027     
18028         for (; i < tokens.length; i++) {
18029           (function(token) {
18030             if (token.type !== 'code') {
18031               return --pending || done();
18032             }
18033             return highlight(token.text, token.lang, function(err, code) {
18034               if (err) { return done(err); }
18035               if (code == null || code === token.text) {
18036                 return --pending || done();
18037               }
18038               token.text = code;
18039               token.escaped = true;
18040               --pending || done();
18041             });
18042           })(tokens[i]);
18043         }
18044     
18045         return;
18046       }
18047       try {
18048         if (opt) { opt = merge({}, marked.defaults, opt); }
18049         return Parser.parse(Lexer.lex(src, opt), opt);
18050       } catch (e) {
18051         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18052         if ((opt || marked.defaults).silent) {
18053           return '<p>An error occured:</p><pre>'
18054             + escape(e.message + '', true)
18055             + '</pre>';
18056         }
18057         throw e;
18058       }
18059     }
18060     
18061     /**
18062      * Options
18063      */
18064     
18065     marked.options =
18066     marked.setOptions = function(opt) {
18067       merge(marked.defaults, opt);
18068       return marked;
18069     };
18070     
18071     marked.defaults = {
18072       gfm: true,
18073       tables: true,
18074       breaks: false,
18075       pedantic: false,
18076       sanitize: false,
18077       sanitizer: null,
18078       mangle: true,
18079       smartLists: false,
18080       silent: false,
18081       highlight: null,
18082       langPrefix: 'lang-',
18083       smartypants: false,
18084       headerPrefix: '',
18085       renderer: new Renderer,
18086       xhtml: false
18087     };
18088     
18089     /**
18090      * Expose
18091      */
18092     
18093     marked.Parser = Parser;
18094     marked.parser = Parser.parse;
18095     
18096     marked.Renderer = Renderer;
18097     
18098     marked.Lexer = Lexer;
18099     marked.lexer = Lexer.lex;
18100     
18101     marked.InlineLexer = InlineLexer;
18102     marked.inlineLexer = InlineLexer.output;
18103     
18104     marked.parse = marked;
18105     
18106     Roo.Markdown.marked = marked;
18107
18108 })();/*
18109  * Based on:
18110  * Ext JS Library 1.1.1
18111  * Copyright(c) 2006-2007, Ext JS, LLC.
18112  *
18113  * Originally Released Under LGPL - original licence link has changed is not relivant.
18114  *
18115  * Fork - LGPL
18116  * <script type="text/javascript">
18117  */
18118
18119
18120
18121 /*
18122  * These classes are derivatives of the similarly named classes in the YUI Library.
18123  * The original license:
18124  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18125  * Code licensed under the BSD License:
18126  * http://developer.yahoo.net/yui/license.txt
18127  */
18128
18129 (function() {
18130
18131 var Event=Roo.EventManager;
18132 var Dom=Roo.lib.Dom;
18133
18134 /**
18135  * @class Roo.dd.DragDrop
18136  * @extends Roo.util.Observable
18137  * Defines the interface and base operation of items that that can be
18138  * dragged or can be drop targets.  It was designed to be extended, overriding
18139  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18140  * Up to three html elements can be associated with a DragDrop instance:
18141  * <ul>
18142  * <li>linked element: the element that is passed into the constructor.
18143  * This is the element which defines the boundaries for interaction with
18144  * other DragDrop objects.</li>
18145  * <li>handle element(s): The drag operation only occurs if the element that
18146  * was clicked matches a handle element.  By default this is the linked
18147  * element, but there are times that you will want only a portion of the
18148  * linked element to initiate the drag operation, and the setHandleElId()
18149  * method provides a way to define this.</li>
18150  * <li>drag element: this represents the element that would be moved along
18151  * with the cursor during a drag operation.  By default, this is the linked
18152  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18153  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18154  * </li>
18155  * </ul>
18156  * This class should not be instantiated until the onload event to ensure that
18157  * the associated elements are available.
18158  * The following would define a DragDrop obj that would interact with any
18159  * other DragDrop obj in the "group1" group:
18160  * <pre>
18161  *  dd = new Roo.dd.DragDrop("div1", "group1");
18162  * </pre>
18163  * Since none of the event handlers have been implemented, nothing would
18164  * actually happen if you were to run the code above.  Normally you would
18165  * override this class or one of the default implementations, but you can
18166  * also override the methods you want on an instance of the class...
18167  * <pre>
18168  *  dd.onDragDrop = function(e, id) {
18169  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18170  *  }
18171  * </pre>
18172  * @constructor
18173  * @param {String} id of the element that is linked to this instance
18174  * @param {String} sGroup the group of related DragDrop objects
18175  * @param {object} config an object containing configurable attributes
18176  *                Valid properties for DragDrop:
18177  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18178  */
18179 Roo.dd.DragDrop = function(id, sGroup, config) {
18180     if (id) {
18181         this.init(id, sGroup, config);
18182     }
18183     
18184 };
18185
18186 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18187
18188     /**
18189      * The id of the element associated with this object.  This is what we
18190      * refer to as the "linked element" because the size and position of
18191      * this element is used to determine when the drag and drop objects have
18192      * interacted.
18193      * @property id
18194      * @type String
18195      */
18196     id: null,
18197
18198     /**
18199      * Configuration attributes passed into the constructor
18200      * @property config
18201      * @type object
18202      */
18203     config: null,
18204
18205     /**
18206      * The id of the element that will be dragged.  By default this is same
18207      * as the linked element , but could be changed to another element. Ex:
18208      * Roo.dd.DDProxy
18209      * @property dragElId
18210      * @type String
18211      * @private
18212      */
18213     dragElId: null,
18214
18215     /**
18216      * the id of the element that initiates the drag operation.  By default
18217      * this is the linked element, but could be changed to be a child of this
18218      * element.  This lets us do things like only starting the drag when the
18219      * header element within the linked html element is clicked.
18220      * @property handleElId
18221      * @type String
18222      * @private
18223      */
18224     handleElId: null,
18225
18226     /**
18227      * An associative array of HTML tags that will be ignored if clicked.
18228      * @property invalidHandleTypes
18229      * @type {string: string}
18230      */
18231     invalidHandleTypes: null,
18232
18233     /**
18234      * An associative array of ids for elements that will be ignored if clicked
18235      * @property invalidHandleIds
18236      * @type {string: string}
18237      */
18238     invalidHandleIds: null,
18239
18240     /**
18241      * An indexted array of css class names for elements that will be ignored
18242      * if clicked.
18243      * @property invalidHandleClasses
18244      * @type string[]
18245      */
18246     invalidHandleClasses: null,
18247
18248     /**
18249      * The linked element's absolute X position at the time the drag was
18250      * started
18251      * @property startPageX
18252      * @type int
18253      * @private
18254      */
18255     startPageX: 0,
18256
18257     /**
18258      * The linked element's absolute X position at the time the drag was
18259      * started
18260      * @property startPageY
18261      * @type int
18262      * @private
18263      */
18264     startPageY: 0,
18265
18266     /**
18267      * The group defines a logical collection of DragDrop objects that are
18268      * related.  Instances only get events when interacting with other
18269      * DragDrop object in the same group.  This lets us define multiple
18270      * groups using a single DragDrop subclass if we want.
18271      * @property groups
18272      * @type {string: string}
18273      */
18274     groups: null,
18275
18276     /**
18277      * Individual drag/drop instances can be locked.  This will prevent
18278      * onmousedown start drag.
18279      * @property locked
18280      * @type boolean
18281      * @private
18282      */
18283     locked: false,
18284
18285     /**
18286      * Lock this instance
18287      * @method lock
18288      */
18289     lock: function() { this.locked = true; },
18290
18291     /**
18292      * Unlock this instace
18293      * @method unlock
18294      */
18295     unlock: function() { this.locked = false; },
18296
18297     /**
18298      * By default, all insances can be a drop target.  This can be disabled by
18299      * setting isTarget to false.
18300      * @method isTarget
18301      * @type boolean
18302      */
18303     isTarget: true,
18304
18305     /**
18306      * The padding configured for this drag and drop object for calculating
18307      * the drop zone intersection with this object.
18308      * @method padding
18309      * @type int[]
18310      */
18311     padding: null,
18312
18313     /**
18314      * Cached reference to the linked element
18315      * @property _domRef
18316      * @private
18317      */
18318     _domRef: null,
18319
18320     /**
18321      * Internal typeof flag
18322      * @property __ygDragDrop
18323      * @private
18324      */
18325     __ygDragDrop: true,
18326
18327     /**
18328      * Set to true when horizontal contraints are applied
18329      * @property constrainX
18330      * @type boolean
18331      * @private
18332      */
18333     constrainX: false,
18334
18335     /**
18336      * Set to true when vertical contraints are applied
18337      * @property constrainY
18338      * @type boolean
18339      * @private
18340      */
18341     constrainY: false,
18342
18343     /**
18344      * The left constraint
18345      * @property minX
18346      * @type int
18347      * @private
18348      */
18349     minX: 0,
18350
18351     /**
18352      * The right constraint
18353      * @property maxX
18354      * @type int
18355      * @private
18356      */
18357     maxX: 0,
18358
18359     /**
18360      * The up constraint
18361      * @property minY
18362      * @type int
18363      * @type int
18364      * @private
18365      */
18366     minY: 0,
18367
18368     /**
18369      * The down constraint
18370      * @property maxY
18371      * @type int
18372      * @private
18373      */
18374     maxY: 0,
18375
18376     /**
18377      * Maintain offsets when we resetconstraints.  Set to true when you want
18378      * the position of the element relative to its parent to stay the same
18379      * when the page changes
18380      *
18381      * @property maintainOffset
18382      * @type boolean
18383      */
18384     maintainOffset: false,
18385
18386     /**
18387      * Array of pixel locations the element will snap to if we specified a
18388      * horizontal graduation/interval.  This array is generated automatically
18389      * when you define a tick interval.
18390      * @property xTicks
18391      * @type int[]
18392      */
18393     xTicks: null,
18394
18395     /**
18396      * Array of pixel locations the element will snap to if we specified a
18397      * vertical graduation/interval.  This array is generated automatically
18398      * when you define a tick interval.
18399      * @property yTicks
18400      * @type int[]
18401      */
18402     yTicks: null,
18403
18404     /**
18405      * By default the drag and drop instance will only respond to the primary
18406      * button click (left button for a right-handed mouse).  Set to true to
18407      * allow drag and drop to start with any mouse click that is propogated
18408      * by the browser
18409      * @property primaryButtonOnly
18410      * @type boolean
18411      */
18412     primaryButtonOnly: true,
18413
18414     /**
18415      * The availabe property is false until the linked dom element is accessible.
18416      * @property available
18417      * @type boolean
18418      */
18419     available: false,
18420
18421     /**
18422      * By default, drags can only be initiated if the mousedown occurs in the
18423      * region the linked element is.  This is done in part to work around a
18424      * bug in some browsers that mis-report the mousedown if the previous
18425      * mouseup happened outside of the window.  This property is set to true
18426      * if outer handles are defined.
18427      *
18428      * @property hasOuterHandles
18429      * @type boolean
18430      * @default false
18431      */
18432     hasOuterHandles: false,
18433
18434     /**
18435      * Code that executes immediately before the startDrag event
18436      * @method b4StartDrag
18437      * @private
18438      */
18439     b4StartDrag: function(x, y) { },
18440
18441     /**
18442      * Abstract method called after a drag/drop object is clicked
18443      * and the drag or mousedown time thresholds have beeen met.
18444      * @method startDrag
18445      * @param {int} X click location
18446      * @param {int} Y click location
18447      */
18448     startDrag: function(x, y) { /* override this */ },
18449
18450     /**
18451      * Code that executes immediately before the onDrag event
18452      * @method b4Drag
18453      * @private
18454      */
18455     b4Drag: function(e) { },
18456
18457     /**
18458      * Abstract method called during the onMouseMove event while dragging an
18459      * object.
18460      * @method onDrag
18461      * @param {Event} e the mousemove event
18462      */
18463     onDrag: function(e) { /* override this */ },
18464
18465     /**
18466      * Abstract method called when this element fist begins hovering over
18467      * another DragDrop obj
18468      * @method onDragEnter
18469      * @param {Event} e the mousemove event
18470      * @param {String|DragDrop[]} id In POINT mode, the element
18471      * id this is hovering over.  In INTERSECT mode, an array of one or more
18472      * dragdrop items being hovered over.
18473      */
18474     onDragEnter: function(e, id) { /* override this */ },
18475
18476     /**
18477      * Code that executes immediately before the onDragOver event
18478      * @method b4DragOver
18479      * @private
18480      */
18481     b4DragOver: function(e) { },
18482
18483     /**
18484      * Abstract method called when this element is hovering over another
18485      * DragDrop obj
18486      * @method onDragOver
18487      * @param {Event} e the mousemove event
18488      * @param {String|DragDrop[]} id In POINT mode, the element
18489      * id this is hovering over.  In INTERSECT mode, an array of dd items
18490      * being hovered over.
18491      */
18492     onDragOver: function(e, id) { /* override this */ },
18493
18494     /**
18495      * Code that executes immediately before the onDragOut event
18496      * @method b4DragOut
18497      * @private
18498      */
18499     b4DragOut: function(e) { },
18500
18501     /**
18502      * Abstract method called when we are no longer hovering over an element
18503      * @method onDragOut
18504      * @param {Event} e the mousemove event
18505      * @param {String|DragDrop[]} id In POINT mode, the element
18506      * id this was hovering over.  In INTERSECT mode, an array of dd items
18507      * that the mouse is no longer over.
18508      */
18509     onDragOut: function(e, id) { /* override this */ },
18510
18511     /**
18512      * Code that executes immediately before the onDragDrop event
18513      * @method b4DragDrop
18514      * @private
18515      */
18516     b4DragDrop: function(e) { },
18517
18518     /**
18519      * Abstract method called when this item is dropped on another DragDrop
18520      * obj
18521      * @method onDragDrop
18522      * @param {Event} e the mouseup event
18523      * @param {String|DragDrop[]} id In POINT mode, the element
18524      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18525      * was dropped on.
18526      */
18527     onDragDrop: function(e, id) { /* override this */ },
18528
18529     /**
18530      * Abstract method called when this item is dropped on an area with no
18531      * drop target
18532      * @method onInvalidDrop
18533      * @param {Event} e the mouseup event
18534      */
18535     onInvalidDrop: function(e) { /* override this */ },
18536
18537     /**
18538      * Code that executes immediately before the endDrag event
18539      * @method b4EndDrag
18540      * @private
18541      */
18542     b4EndDrag: function(e) { },
18543
18544     /**
18545      * Fired when we are done dragging the object
18546      * @method endDrag
18547      * @param {Event} e the mouseup event
18548      */
18549     endDrag: function(e) { /* override this */ },
18550
18551     /**
18552      * Code executed immediately before the onMouseDown event
18553      * @method b4MouseDown
18554      * @param {Event} e the mousedown event
18555      * @private
18556      */
18557     b4MouseDown: function(e) {  },
18558
18559     /**
18560      * Event handler that fires when a drag/drop obj gets a mousedown
18561      * @method onMouseDown
18562      * @param {Event} e the mousedown event
18563      */
18564     onMouseDown: function(e) { /* override this */ },
18565
18566     /**
18567      * Event handler that fires when a drag/drop obj gets a mouseup
18568      * @method onMouseUp
18569      * @param {Event} e the mouseup event
18570      */
18571     onMouseUp: function(e) { /* override this */ },
18572
18573     /**
18574      * Override the onAvailable method to do what is needed after the initial
18575      * position was determined.
18576      * @method onAvailable
18577      */
18578     onAvailable: function () {
18579     },
18580
18581     /*
18582      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18583      * @type Object
18584      */
18585     defaultPadding : {left:0, right:0, top:0, bottom:0},
18586
18587     /*
18588      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18589  *
18590  * Usage:
18591  <pre><code>
18592  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18593                 { dragElId: "existingProxyDiv" });
18594  dd.startDrag = function(){
18595      this.constrainTo("parent-id");
18596  };
18597  </code></pre>
18598  * Or you can initalize it using the {@link Roo.Element} object:
18599  <pre><code>
18600  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18601      startDrag : function(){
18602          this.constrainTo("parent-id");
18603      }
18604  });
18605  </code></pre>
18606      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18607      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18608      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18609      * an object containing the sides to pad. For example: {right:10, bottom:10}
18610      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18611      */
18612     constrainTo : function(constrainTo, pad, inContent){
18613         if(typeof pad == "number"){
18614             pad = {left: pad, right:pad, top:pad, bottom:pad};
18615         }
18616         pad = pad || this.defaultPadding;
18617         var b = Roo.get(this.getEl()).getBox();
18618         var ce = Roo.get(constrainTo);
18619         var s = ce.getScroll();
18620         var c, cd = ce.dom;
18621         if(cd == document.body){
18622             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18623         }else{
18624             xy = ce.getXY();
18625             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18626         }
18627
18628
18629         var topSpace = b.y - c.y;
18630         var leftSpace = b.x - c.x;
18631
18632         this.resetConstraints();
18633         this.setXConstraint(leftSpace - (pad.left||0), // left
18634                 c.width - leftSpace - b.width - (pad.right||0) //right
18635         );
18636         this.setYConstraint(topSpace - (pad.top||0), //top
18637                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18638         );
18639     },
18640
18641     /**
18642      * Returns a reference to the linked element
18643      * @method getEl
18644      * @return {HTMLElement} the html element
18645      */
18646     getEl: function() {
18647         if (!this._domRef) {
18648             this._domRef = Roo.getDom(this.id);
18649         }
18650
18651         return this._domRef;
18652     },
18653
18654     /**
18655      * Returns a reference to the actual element to drag.  By default this is
18656      * the same as the html element, but it can be assigned to another
18657      * element. An example of this can be found in Roo.dd.DDProxy
18658      * @method getDragEl
18659      * @return {HTMLElement} the html element
18660      */
18661     getDragEl: function() {
18662         return Roo.getDom(this.dragElId);
18663     },
18664
18665     /**
18666      * Sets up the DragDrop object.  Must be called in the constructor of any
18667      * Roo.dd.DragDrop subclass
18668      * @method init
18669      * @param id the id of the linked element
18670      * @param {String} sGroup the group of related items
18671      * @param {object} config configuration attributes
18672      */
18673     init: function(id, sGroup, config) {
18674         this.initTarget(id, sGroup, config);
18675         if (!Roo.isTouch) {
18676             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18677         }
18678         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18679         // Event.on(this.id, "selectstart", Event.preventDefault);
18680     },
18681
18682     /**
18683      * Initializes Targeting functionality only... the object does not
18684      * get a mousedown handler.
18685      * @method initTarget
18686      * @param id the id of the linked element
18687      * @param {String} sGroup the group of related items
18688      * @param {object} config configuration attributes
18689      */
18690     initTarget: function(id, sGroup, config) {
18691
18692         // configuration attributes
18693         this.config = config || {};
18694
18695         // create a local reference to the drag and drop manager
18696         this.DDM = Roo.dd.DDM;
18697         // initialize the groups array
18698         this.groups = {};
18699
18700         // assume that we have an element reference instead of an id if the
18701         // parameter is not a string
18702         if (typeof id !== "string") {
18703             id = Roo.id(id);
18704         }
18705
18706         // set the id
18707         this.id = id;
18708
18709         // add to an interaction group
18710         this.addToGroup((sGroup) ? sGroup : "default");
18711
18712         // We don't want to register this as the handle with the manager
18713         // so we just set the id rather than calling the setter.
18714         this.handleElId = id;
18715
18716         // the linked element is the element that gets dragged by default
18717         this.setDragElId(id);
18718
18719         // by default, clicked anchors will not start drag operations.
18720         this.invalidHandleTypes = { A: "A" };
18721         this.invalidHandleIds = {};
18722         this.invalidHandleClasses = [];
18723
18724         this.applyConfig();
18725
18726         this.handleOnAvailable();
18727     },
18728
18729     /**
18730      * Applies the configuration parameters that were passed into the constructor.
18731      * This is supposed to happen at each level through the inheritance chain.  So
18732      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18733      * DragDrop in order to get all of the parameters that are available in
18734      * each object.
18735      * @method applyConfig
18736      */
18737     applyConfig: function() {
18738
18739         // configurable properties:
18740         //    padding, isTarget, maintainOffset, primaryButtonOnly
18741         this.padding           = this.config.padding || [0, 0, 0, 0];
18742         this.isTarget          = (this.config.isTarget !== false);
18743         this.maintainOffset    = (this.config.maintainOffset);
18744         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18745
18746     },
18747
18748     /**
18749      * Executed when the linked element is available
18750      * @method handleOnAvailable
18751      * @private
18752      */
18753     handleOnAvailable: function() {
18754         this.available = true;
18755         this.resetConstraints();
18756         this.onAvailable();
18757     },
18758
18759      /**
18760      * Configures the padding for the target zone in px.  Effectively expands
18761      * (or reduces) the virtual object size for targeting calculations.
18762      * Supports css-style shorthand; if only one parameter is passed, all sides
18763      * will have that padding, and if only two are passed, the top and bottom
18764      * will have the first param, the left and right the second.
18765      * @method setPadding
18766      * @param {int} iTop    Top pad
18767      * @param {int} iRight  Right pad
18768      * @param {int} iBot    Bot pad
18769      * @param {int} iLeft   Left pad
18770      */
18771     setPadding: function(iTop, iRight, iBot, iLeft) {
18772         // this.padding = [iLeft, iRight, iTop, iBot];
18773         if (!iRight && 0 !== iRight) {
18774             this.padding = [iTop, iTop, iTop, iTop];
18775         } else if (!iBot && 0 !== iBot) {
18776             this.padding = [iTop, iRight, iTop, iRight];
18777         } else {
18778             this.padding = [iTop, iRight, iBot, iLeft];
18779         }
18780     },
18781
18782     /**
18783      * Stores the initial placement of the linked element.
18784      * @method setInitialPosition
18785      * @param {int} diffX   the X offset, default 0
18786      * @param {int} diffY   the Y offset, default 0
18787      */
18788     setInitPosition: function(diffX, diffY) {
18789         var el = this.getEl();
18790
18791         if (!this.DDM.verifyEl(el)) {
18792             return;
18793         }
18794
18795         var dx = diffX || 0;
18796         var dy = diffY || 0;
18797
18798         var p = Dom.getXY( el );
18799
18800         this.initPageX = p[0] - dx;
18801         this.initPageY = p[1] - dy;
18802
18803         this.lastPageX = p[0];
18804         this.lastPageY = p[1];
18805
18806
18807         this.setStartPosition(p);
18808     },
18809
18810     /**
18811      * Sets the start position of the element.  This is set when the obj
18812      * is initialized, the reset when a drag is started.
18813      * @method setStartPosition
18814      * @param pos current position (from previous lookup)
18815      * @private
18816      */
18817     setStartPosition: function(pos) {
18818         var p = pos || Dom.getXY( this.getEl() );
18819         this.deltaSetXY = null;
18820
18821         this.startPageX = p[0];
18822         this.startPageY = p[1];
18823     },
18824
18825     /**
18826      * Add this instance to a group of related drag/drop objects.  All
18827      * instances belong to at least one group, and can belong to as many
18828      * groups as needed.
18829      * @method addToGroup
18830      * @param sGroup {string} the name of the group
18831      */
18832     addToGroup: function(sGroup) {
18833         this.groups[sGroup] = true;
18834         this.DDM.regDragDrop(this, sGroup);
18835     },
18836
18837     /**
18838      * Remove's this instance from the supplied interaction group
18839      * @method removeFromGroup
18840      * @param {string}  sGroup  The group to drop
18841      */
18842     removeFromGroup: function(sGroup) {
18843         if (this.groups[sGroup]) {
18844             delete this.groups[sGroup];
18845         }
18846
18847         this.DDM.removeDDFromGroup(this, sGroup);
18848     },
18849
18850     /**
18851      * Allows you to specify that an element other than the linked element
18852      * will be moved with the cursor during a drag
18853      * @method setDragElId
18854      * @param id {string} the id of the element that will be used to initiate the drag
18855      */
18856     setDragElId: function(id) {
18857         this.dragElId = id;
18858     },
18859
18860     /**
18861      * Allows you to specify a child of the linked element that should be
18862      * used to initiate the drag operation.  An example of this would be if
18863      * you have a content div with text and links.  Clicking anywhere in the
18864      * content area would normally start the drag operation.  Use this method
18865      * to specify that an element inside of the content div is the element
18866      * that starts the drag operation.
18867      * @method setHandleElId
18868      * @param id {string} the id of the element that will be used to
18869      * initiate the drag.
18870      */
18871     setHandleElId: function(id) {
18872         if (typeof id !== "string") {
18873             id = Roo.id(id);
18874         }
18875         this.handleElId = id;
18876         this.DDM.regHandle(this.id, id);
18877     },
18878
18879     /**
18880      * Allows you to set an element outside of the linked element as a drag
18881      * handle
18882      * @method setOuterHandleElId
18883      * @param id the id of the element that will be used to initiate the drag
18884      */
18885     setOuterHandleElId: function(id) {
18886         if (typeof id !== "string") {
18887             id = Roo.id(id);
18888         }
18889         Event.on(id, "mousedown",
18890                 this.handleMouseDown, this);
18891         this.setHandleElId(id);
18892
18893         this.hasOuterHandles = true;
18894     },
18895
18896     /**
18897      * Remove all drag and drop hooks for this element
18898      * @method unreg
18899      */
18900     unreg: function() {
18901         Event.un(this.id, "mousedown",
18902                 this.handleMouseDown);
18903         Event.un(this.id, "touchstart",
18904                 this.handleMouseDown);
18905         this._domRef = null;
18906         this.DDM._remove(this);
18907     },
18908
18909     destroy : function(){
18910         this.unreg();
18911     },
18912
18913     /**
18914      * Returns true if this instance is locked, or the drag drop mgr is locked
18915      * (meaning that all drag/drop is disabled on the page.)
18916      * @method isLocked
18917      * @return {boolean} true if this obj or all drag/drop is locked, else
18918      * false
18919      */
18920     isLocked: function() {
18921         return (this.DDM.isLocked() || this.locked);
18922     },
18923
18924     /**
18925      * Fired when this object is clicked
18926      * @method handleMouseDown
18927      * @param {Event} e
18928      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18929      * @private
18930      */
18931     handleMouseDown: function(e, oDD){
18932      
18933         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18934             //Roo.log('not touch/ button !=0');
18935             return;
18936         }
18937         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18938             return; // double touch..
18939         }
18940         
18941
18942         if (this.isLocked()) {
18943             //Roo.log('locked');
18944             return;
18945         }
18946
18947         this.DDM.refreshCache(this.groups);
18948 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18949         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18950         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18951             //Roo.log('no outer handes or not over target');
18952                 // do nothing.
18953         } else {
18954 //            Roo.log('check validator');
18955             if (this.clickValidator(e)) {
18956 //                Roo.log('validate success');
18957                 // set the initial element position
18958                 this.setStartPosition();
18959
18960
18961                 this.b4MouseDown(e);
18962                 this.onMouseDown(e);
18963
18964                 this.DDM.handleMouseDown(e, this);
18965
18966                 this.DDM.stopEvent(e);
18967             } else {
18968
18969
18970             }
18971         }
18972     },
18973
18974     clickValidator: function(e) {
18975         var target = e.getTarget();
18976         return ( this.isValidHandleChild(target) &&
18977                     (this.id == this.handleElId ||
18978                         this.DDM.handleWasClicked(target, this.id)) );
18979     },
18980
18981     /**
18982      * Allows you to specify a tag name that should not start a drag operation
18983      * when clicked.  This is designed to facilitate embedding links within a
18984      * drag handle that do something other than start the drag.
18985      * @method addInvalidHandleType
18986      * @param {string} tagName the type of element to exclude
18987      */
18988     addInvalidHandleType: function(tagName) {
18989         var type = tagName.toUpperCase();
18990         this.invalidHandleTypes[type] = type;
18991     },
18992
18993     /**
18994      * Lets you to specify an element id for a child of a drag handle
18995      * that should not initiate a drag
18996      * @method addInvalidHandleId
18997      * @param {string} id the element id of the element you wish to ignore
18998      */
18999     addInvalidHandleId: function(id) {
19000         if (typeof id !== "string") {
19001             id = Roo.id(id);
19002         }
19003         this.invalidHandleIds[id] = id;
19004     },
19005
19006     /**
19007      * Lets you specify a css class of elements that will not initiate a drag
19008      * @method addInvalidHandleClass
19009      * @param {string} cssClass the class of the elements you wish to ignore
19010      */
19011     addInvalidHandleClass: function(cssClass) {
19012         this.invalidHandleClasses.push(cssClass);
19013     },
19014
19015     /**
19016      * Unsets an excluded tag name set by addInvalidHandleType
19017      * @method removeInvalidHandleType
19018      * @param {string} tagName the type of element to unexclude
19019      */
19020     removeInvalidHandleType: function(tagName) {
19021         var type = tagName.toUpperCase();
19022         // this.invalidHandleTypes[type] = null;
19023         delete this.invalidHandleTypes[type];
19024     },
19025
19026     /**
19027      * Unsets an invalid handle id
19028      * @method removeInvalidHandleId
19029      * @param {string} id the id of the element to re-enable
19030      */
19031     removeInvalidHandleId: function(id) {
19032         if (typeof id !== "string") {
19033             id = Roo.id(id);
19034         }
19035         delete this.invalidHandleIds[id];
19036     },
19037
19038     /**
19039      * Unsets an invalid css class
19040      * @method removeInvalidHandleClass
19041      * @param {string} cssClass the class of the element(s) you wish to
19042      * re-enable
19043      */
19044     removeInvalidHandleClass: function(cssClass) {
19045         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19046             if (this.invalidHandleClasses[i] == cssClass) {
19047                 delete this.invalidHandleClasses[i];
19048             }
19049         }
19050     },
19051
19052     /**
19053      * Checks the tag exclusion list to see if this click should be ignored
19054      * @method isValidHandleChild
19055      * @param {HTMLElement} node the HTMLElement to evaluate
19056      * @return {boolean} true if this is a valid tag type, false if not
19057      */
19058     isValidHandleChild: function(node) {
19059
19060         var valid = true;
19061         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19062         var nodeName;
19063         try {
19064             nodeName = node.nodeName.toUpperCase();
19065         } catch(e) {
19066             nodeName = node.nodeName;
19067         }
19068         valid = valid && !this.invalidHandleTypes[nodeName];
19069         valid = valid && !this.invalidHandleIds[node.id];
19070
19071         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19072             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19073         }
19074
19075
19076         return valid;
19077
19078     },
19079
19080     /**
19081      * Create the array of horizontal tick marks if an interval was specified
19082      * in setXConstraint().
19083      * @method setXTicks
19084      * @private
19085      */
19086     setXTicks: function(iStartX, iTickSize) {
19087         this.xTicks = [];
19088         this.xTickSize = iTickSize;
19089
19090         var tickMap = {};
19091
19092         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19093             if (!tickMap[i]) {
19094                 this.xTicks[this.xTicks.length] = i;
19095                 tickMap[i] = true;
19096             }
19097         }
19098
19099         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19100             if (!tickMap[i]) {
19101                 this.xTicks[this.xTicks.length] = i;
19102                 tickMap[i] = true;
19103             }
19104         }
19105
19106         this.xTicks.sort(this.DDM.numericSort) ;
19107     },
19108
19109     /**
19110      * Create the array of vertical tick marks if an interval was specified in
19111      * setYConstraint().
19112      * @method setYTicks
19113      * @private
19114      */
19115     setYTicks: function(iStartY, iTickSize) {
19116         this.yTicks = [];
19117         this.yTickSize = iTickSize;
19118
19119         var tickMap = {};
19120
19121         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19122             if (!tickMap[i]) {
19123                 this.yTicks[this.yTicks.length] = i;
19124                 tickMap[i] = true;
19125             }
19126         }
19127
19128         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19129             if (!tickMap[i]) {
19130                 this.yTicks[this.yTicks.length] = i;
19131                 tickMap[i] = true;
19132             }
19133         }
19134
19135         this.yTicks.sort(this.DDM.numericSort) ;
19136     },
19137
19138     /**
19139      * By default, the element can be dragged any place on the screen.  Use
19140      * this method to limit the horizontal travel of the element.  Pass in
19141      * 0,0 for the parameters if you want to lock the drag to the y axis.
19142      * @method setXConstraint
19143      * @param {int} iLeft the number of pixels the element can move to the left
19144      * @param {int} iRight the number of pixels the element can move to the
19145      * right
19146      * @param {int} iTickSize optional parameter for specifying that the
19147      * element
19148      * should move iTickSize pixels at a time.
19149      */
19150     setXConstraint: function(iLeft, iRight, iTickSize) {
19151         this.leftConstraint = iLeft;
19152         this.rightConstraint = iRight;
19153
19154         this.minX = this.initPageX - iLeft;
19155         this.maxX = this.initPageX + iRight;
19156         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19157
19158         this.constrainX = true;
19159     },
19160
19161     /**
19162      * Clears any constraints applied to this instance.  Also clears ticks
19163      * since they can't exist independent of a constraint at this time.
19164      * @method clearConstraints
19165      */
19166     clearConstraints: function() {
19167         this.constrainX = false;
19168         this.constrainY = false;
19169         this.clearTicks();
19170     },
19171
19172     /**
19173      * Clears any tick interval defined for this instance
19174      * @method clearTicks
19175      */
19176     clearTicks: function() {
19177         this.xTicks = null;
19178         this.yTicks = null;
19179         this.xTickSize = 0;
19180         this.yTickSize = 0;
19181     },
19182
19183     /**
19184      * By default, the element can be dragged any place on the screen.  Set
19185      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19186      * parameters if you want to lock the drag to the x axis.
19187      * @method setYConstraint
19188      * @param {int} iUp the number of pixels the element can move up
19189      * @param {int} iDown the number of pixels the element can move down
19190      * @param {int} iTickSize optional parameter for specifying that the
19191      * element should move iTickSize pixels at a time.
19192      */
19193     setYConstraint: function(iUp, iDown, iTickSize) {
19194         this.topConstraint = iUp;
19195         this.bottomConstraint = iDown;
19196
19197         this.minY = this.initPageY - iUp;
19198         this.maxY = this.initPageY + iDown;
19199         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19200
19201         this.constrainY = true;
19202
19203     },
19204
19205     /**
19206      * resetConstraints must be called if you manually reposition a dd element.
19207      * @method resetConstraints
19208      * @param {boolean} maintainOffset
19209      */
19210     resetConstraints: function() {
19211
19212
19213         // Maintain offsets if necessary
19214         if (this.initPageX || this.initPageX === 0) {
19215             // figure out how much this thing has moved
19216             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19217             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19218
19219             this.setInitPosition(dx, dy);
19220
19221         // This is the first time we have detected the element's position
19222         } else {
19223             this.setInitPosition();
19224         }
19225
19226         if (this.constrainX) {
19227             this.setXConstraint( this.leftConstraint,
19228                                  this.rightConstraint,
19229                                  this.xTickSize        );
19230         }
19231
19232         if (this.constrainY) {
19233             this.setYConstraint( this.topConstraint,
19234                                  this.bottomConstraint,
19235                                  this.yTickSize         );
19236         }
19237     },
19238
19239     /**
19240      * Normally the drag element is moved pixel by pixel, but we can specify
19241      * that it move a number of pixels at a time.  This method resolves the
19242      * location when we have it set up like this.
19243      * @method getTick
19244      * @param {int} val where we want to place the object
19245      * @param {int[]} tickArray sorted array of valid points
19246      * @return {int} the closest tick
19247      * @private
19248      */
19249     getTick: function(val, tickArray) {
19250
19251         if (!tickArray) {
19252             // If tick interval is not defined, it is effectively 1 pixel,
19253             // so we return the value passed to us.
19254             return val;
19255         } else if (tickArray[0] >= val) {
19256             // The value is lower than the first tick, so we return the first
19257             // tick.
19258             return tickArray[0];
19259         } else {
19260             for (var i=0, len=tickArray.length; i<len; ++i) {
19261                 var next = i + 1;
19262                 if (tickArray[next] && tickArray[next] >= val) {
19263                     var diff1 = val - tickArray[i];
19264                     var diff2 = tickArray[next] - val;
19265                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19266                 }
19267             }
19268
19269             // The value is larger than the last tick, so we return the last
19270             // tick.
19271             return tickArray[tickArray.length - 1];
19272         }
19273     },
19274
19275     /**
19276      * toString method
19277      * @method toString
19278      * @return {string} string representation of the dd obj
19279      */
19280     toString: function() {
19281         return ("DragDrop " + this.id);
19282     }
19283
19284 });
19285
19286 })();
19287 /*
19288  * Based on:
19289  * Ext JS Library 1.1.1
19290  * Copyright(c) 2006-2007, Ext JS, LLC.
19291  *
19292  * Originally Released Under LGPL - original licence link has changed is not relivant.
19293  *
19294  * Fork - LGPL
19295  * <script type="text/javascript">
19296  */
19297
19298
19299 /**
19300  * The drag and drop utility provides a framework for building drag and drop
19301  * applications.  In addition to enabling drag and drop for specific elements,
19302  * the drag and drop elements are tracked by the manager class, and the
19303  * interactions between the various elements are tracked during the drag and
19304  * the implementing code is notified about these important moments.
19305  */
19306
19307 // Only load the library once.  Rewriting the manager class would orphan
19308 // existing drag and drop instances.
19309 if (!Roo.dd.DragDropMgr) {
19310
19311 /**
19312  * @class Roo.dd.DragDropMgr
19313  * DragDropMgr is a singleton that tracks the element interaction for
19314  * all DragDrop items in the window.  Generally, you will not call
19315  * this class directly, but it does have helper methods that could
19316  * be useful in your DragDrop implementations.
19317  * @singleton
19318  */
19319 Roo.dd.DragDropMgr = function() {
19320
19321     var Event = Roo.EventManager;
19322
19323     return {
19324
19325         /**
19326          * Two dimensional Array of registered DragDrop objects.  The first
19327          * dimension is the DragDrop item group, the second the DragDrop
19328          * object.
19329          * @property ids
19330          * @type {string: string}
19331          * @private
19332          * @static
19333          */
19334         ids: {},
19335
19336         /**
19337          * Array of element ids defined as drag handles.  Used to determine
19338          * if the element that generated the mousedown event is actually the
19339          * handle and not the html element itself.
19340          * @property handleIds
19341          * @type {string: string}
19342          * @private
19343          * @static
19344          */
19345         handleIds: {},
19346
19347         /**
19348          * the DragDrop object that is currently being dragged
19349          * @property dragCurrent
19350          * @type DragDrop
19351          * @private
19352          * @static
19353          **/
19354         dragCurrent: null,
19355
19356         /**
19357          * the DragDrop object(s) that are being hovered over
19358          * @property dragOvers
19359          * @type Array
19360          * @private
19361          * @static
19362          */
19363         dragOvers: {},
19364
19365         /**
19366          * the X distance between the cursor and the object being dragged
19367          * @property deltaX
19368          * @type int
19369          * @private
19370          * @static
19371          */
19372         deltaX: 0,
19373
19374         /**
19375          * the Y distance between the cursor and the object being dragged
19376          * @property deltaY
19377          * @type int
19378          * @private
19379          * @static
19380          */
19381         deltaY: 0,
19382
19383         /**
19384          * Flag to determine if we should prevent the default behavior of the
19385          * events we define. By default this is true, but this can be set to
19386          * false if you need the default behavior (not recommended)
19387          * @property preventDefault
19388          * @type boolean
19389          * @static
19390          */
19391         preventDefault: true,
19392
19393         /**
19394          * Flag to determine if we should stop the propagation of the events
19395          * we generate. This is true by default but you may want to set it to
19396          * false if the html element contains other features that require the
19397          * mouse click.
19398          * @property stopPropagation
19399          * @type boolean
19400          * @static
19401          */
19402         stopPropagation: true,
19403
19404         /**
19405          * Internal flag that is set to true when drag and drop has been
19406          * intialized
19407          * @property initialized
19408          * @private
19409          * @static
19410          */
19411         initalized: false,
19412
19413         /**
19414          * All drag and drop can be disabled.
19415          * @property locked
19416          * @private
19417          * @static
19418          */
19419         locked: false,
19420
19421         /**
19422          * Called the first time an element is registered.
19423          * @method init
19424          * @private
19425          * @static
19426          */
19427         init: function() {
19428             this.initialized = true;
19429         },
19430
19431         /**
19432          * In point mode, drag and drop interaction is defined by the
19433          * location of the cursor during the drag/drop
19434          * @property POINT
19435          * @type int
19436          * @static
19437          */
19438         POINT: 0,
19439
19440         /**
19441          * In intersect mode, drag and drop interactio nis defined by the
19442          * overlap of two or more drag and drop objects.
19443          * @property INTERSECT
19444          * @type int
19445          * @static
19446          */
19447         INTERSECT: 1,
19448
19449         /**
19450          * The current drag and drop mode.  Default: POINT
19451          * @property mode
19452          * @type int
19453          * @static
19454          */
19455         mode: 0,
19456
19457         /**
19458          * Runs method on all drag and drop objects
19459          * @method _execOnAll
19460          * @private
19461          * @static
19462          */
19463         _execOnAll: function(sMethod, args) {
19464             for (var i in this.ids) {
19465                 for (var j in this.ids[i]) {
19466                     var oDD = this.ids[i][j];
19467                     if (! this.isTypeOfDD(oDD)) {
19468                         continue;
19469                     }
19470                     oDD[sMethod].apply(oDD, args);
19471                 }
19472             }
19473         },
19474
19475         /**
19476          * Drag and drop initialization.  Sets up the global event handlers
19477          * @method _onLoad
19478          * @private
19479          * @static
19480          */
19481         _onLoad: function() {
19482
19483             this.init();
19484
19485             if (!Roo.isTouch) {
19486                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19487                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19488             }
19489             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19490             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19491             
19492             Event.on(window,   "unload",    this._onUnload, this, true);
19493             Event.on(window,   "resize",    this._onResize, this, true);
19494             // Event.on(window,   "mouseout",    this._test);
19495
19496         },
19497
19498         /**
19499          * Reset constraints on all drag and drop objs
19500          * @method _onResize
19501          * @private
19502          * @static
19503          */
19504         _onResize: function(e) {
19505             this._execOnAll("resetConstraints", []);
19506         },
19507
19508         /**
19509          * Lock all drag and drop functionality
19510          * @method lock
19511          * @static
19512          */
19513         lock: function() { this.locked = true; },
19514
19515         /**
19516          * Unlock all drag and drop functionality
19517          * @method unlock
19518          * @static
19519          */
19520         unlock: function() { this.locked = false; },
19521
19522         /**
19523          * Is drag and drop locked?
19524          * @method isLocked
19525          * @return {boolean} True if drag and drop is locked, false otherwise.
19526          * @static
19527          */
19528         isLocked: function() { return this.locked; },
19529
19530         /**
19531          * Location cache that is set for all drag drop objects when a drag is
19532          * initiated, cleared when the drag is finished.
19533          * @property locationCache
19534          * @private
19535          * @static
19536          */
19537         locationCache: {},
19538
19539         /**
19540          * Set useCache to false if you want to force object the lookup of each
19541          * drag and drop linked element constantly during a drag.
19542          * @property useCache
19543          * @type boolean
19544          * @static
19545          */
19546         useCache: true,
19547
19548         /**
19549          * The number of pixels that the mouse needs to move after the
19550          * mousedown before the drag is initiated.  Default=3;
19551          * @property clickPixelThresh
19552          * @type int
19553          * @static
19554          */
19555         clickPixelThresh: 3,
19556
19557         /**
19558          * The number of milliseconds after the mousedown event to initiate the
19559          * drag if we don't get a mouseup event. Default=1000
19560          * @property clickTimeThresh
19561          * @type int
19562          * @static
19563          */
19564         clickTimeThresh: 350,
19565
19566         /**
19567          * Flag that indicates that either the drag pixel threshold or the
19568          * mousdown time threshold has been met
19569          * @property dragThreshMet
19570          * @type boolean
19571          * @private
19572          * @static
19573          */
19574         dragThreshMet: false,
19575
19576         /**
19577          * Timeout used for the click time threshold
19578          * @property clickTimeout
19579          * @type Object
19580          * @private
19581          * @static
19582          */
19583         clickTimeout: null,
19584
19585         /**
19586          * The X position of the mousedown event stored for later use when a
19587          * drag threshold is met.
19588          * @property startX
19589          * @type int
19590          * @private
19591          * @static
19592          */
19593         startX: 0,
19594
19595         /**
19596          * The Y position of the mousedown event stored for later use when a
19597          * drag threshold is met.
19598          * @property startY
19599          * @type int
19600          * @private
19601          * @static
19602          */
19603         startY: 0,
19604
19605         /**
19606          * Each DragDrop instance must be registered with the DragDropMgr.
19607          * This is executed in DragDrop.init()
19608          * @method regDragDrop
19609          * @param {DragDrop} oDD the DragDrop object to register
19610          * @param {String} sGroup the name of the group this element belongs to
19611          * @static
19612          */
19613         regDragDrop: function(oDD, sGroup) {
19614             if (!this.initialized) { this.init(); }
19615
19616             if (!this.ids[sGroup]) {
19617                 this.ids[sGroup] = {};
19618             }
19619             this.ids[sGroup][oDD.id] = oDD;
19620         },
19621
19622         /**
19623          * Removes the supplied dd instance from the supplied group. Executed
19624          * by DragDrop.removeFromGroup, so don't call this function directly.
19625          * @method removeDDFromGroup
19626          * @private
19627          * @static
19628          */
19629         removeDDFromGroup: function(oDD, sGroup) {
19630             if (!this.ids[sGroup]) {
19631                 this.ids[sGroup] = {};
19632             }
19633
19634             var obj = this.ids[sGroup];
19635             if (obj && obj[oDD.id]) {
19636                 delete obj[oDD.id];
19637             }
19638         },
19639
19640         /**
19641          * Unregisters a drag and drop item.  This is executed in
19642          * DragDrop.unreg, use that method instead of calling this directly.
19643          * @method _remove
19644          * @private
19645          * @static
19646          */
19647         _remove: function(oDD) {
19648             for (var g in oDD.groups) {
19649                 if (g && this.ids[g][oDD.id]) {
19650                     delete this.ids[g][oDD.id];
19651                 }
19652             }
19653             delete this.handleIds[oDD.id];
19654         },
19655
19656         /**
19657          * Each DragDrop handle element must be registered.  This is done
19658          * automatically when executing DragDrop.setHandleElId()
19659          * @method regHandle
19660          * @param {String} sDDId the DragDrop id this element is a handle for
19661          * @param {String} sHandleId the id of the element that is the drag
19662          * handle
19663          * @static
19664          */
19665         regHandle: function(sDDId, sHandleId) {
19666             if (!this.handleIds[sDDId]) {
19667                 this.handleIds[sDDId] = {};
19668             }
19669             this.handleIds[sDDId][sHandleId] = sHandleId;
19670         },
19671
19672         /**
19673          * Utility function to determine if a given element has been
19674          * registered as a drag drop item.
19675          * @method isDragDrop
19676          * @param {String} id the element id to check
19677          * @return {boolean} true if this element is a DragDrop item,
19678          * false otherwise
19679          * @static
19680          */
19681         isDragDrop: function(id) {
19682             return ( this.getDDById(id) ) ? true : false;
19683         },
19684
19685         /**
19686          * Returns the drag and drop instances that are in all groups the
19687          * passed in instance belongs to.
19688          * @method getRelated
19689          * @param {DragDrop} p_oDD the obj to get related data for
19690          * @param {boolean} bTargetsOnly if true, only return targetable objs
19691          * @return {DragDrop[]} the related instances
19692          * @static
19693          */
19694         getRelated: function(p_oDD, bTargetsOnly) {
19695             var oDDs = [];
19696             for (var i in p_oDD.groups) {
19697                 for (j in this.ids[i]) {
19698                     var dd = this.ids[i][j];
19699                     if (! this.isTypeOfDD(dd)) {
19700                         continue;
19701                     }
19702                     if (!bTargetsOnly || dd.isTarget) {
19703                         oDDs[oDDs.length] = dd;
19704                     }
19705                 }
19706             }
19707
19708             return oDDs;
19709         },
19710
19711         /**
19712          * Returns true if the specified dd target is a legal target for
19713          * the specifice drag obj
19714          * @method isLegalTarget
19715          * @param {DragDrop} the drag obj
19716          * @param {DragDrop} the target
19717          * @return {boolean} true if the target is a legal target for the
19718          * dd obj
19719          * @static
19720          */
19721         isLegalTarget: function (oDD, oTargetDD) {
19722             var targets = this.getRelated(oDD, true);
19723             for (var i=0, len=targets.length;i<len;++i) {
19724                 if (targets[i].id == oTargetDD.id) {
19725                     return true;
19726                 }
19727             }
19728
19729             return false;
19730         },
19731
19732         /**
19733          * My goal is to be able to transparently determine if an object is
19734          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19735          * returns "object", oDD.constructor.toString() always returns
19736          * "DragDrop" and not the name of the subclass.  So for now it just
19737          * evaluates a well-known variable in DragDrop.
19738          * @method isTypeOfDD
19739          * @param {Object} the object to evaluate
19740          * @return {boolean} true if typeof oDD = DragDrop
19741          * @static
19742          */
19743         isTypeOfDD: function (oDD) {
19744             return (oDD && oDD.__ygDragDrop);
19745         },
19746
19747         /**
19748          * Utility function to determine if a given element has been
19749          * registered as a drag drop handle for the given Drag Drop object.
19750          * @method isHandle
19751          * @param {String} id the element id to check
19752          * @return {boolean} true if this element is a DragDrop handle, false
19753          * otherwise
19754          * @static
19755          */
19756         isHandle: function(sDDId, sHandleId) {
19757             return ( this.handleIds[sDDId] &&
19758                             this.handleIds[sDDId][sHandleId] );
19759         },
19760
19761         /**
19762          * Returns the DragDrop instance for a given id
19763          * @method getDDById
19764          * @param {String} id the id of the DragDrop object
19765          * @return {DragDrop} the drag drop object, null if it is not found
19766          * @static
19767          */
19768         getDDById: function(id) {
19769             for (var i in this.ids) {
19770                 if (this.ids[i][id]) {
19771                     return this.ids[i][id];
19772                 }
19773             }
19774             return null;
19775         },
19776
19777         /**
19778          * Fired after a registered DragDrop object gets the mousedown event.
19779          * Sets up the events required to track the object being dragged
19780          * @method handleMouseDown
19781          * @param {Event} e the event
19782          * @param oDD the DragDrop object being dragged
19783          * @private
19784          * @static
19785          */
19786         handleMouseDown: function(e, oDD) {
19787             if(Roo.QuickTips){
19788                 Roo.QuickTips.disable();
19789             }
19790             this.currentTarget = e.getTarget();
19791
19792             this.dragCurrent = oDD;
19793
19794             var el = oDD.getEl();
19795
19796             // track start position
19797             this.startX = e.getPageX();
19798             this.startY = e.getPageY();
19799
19800             this.deltaX = this.startX - el.offsetLeft;
19801             this.deltaY = this.startY - el.offsetTop;
19802
19803             this.dragThreshMet = false;
19804
19805             this.clickTimeout = setTimeout(
19806                     function() {
19807                         var DDM = Roo.dd.DDM;
19808                         DDM.startDrag(DDM.startX, DDM.startY);
19809                     },
19810                     this.clickTimeThresh );
19811         },
19812
19813         /**
19814          * Fired when either the drag pixel threshol or the mousedown hold
19815          * time threshold has been met.
19816          * @method startDrag
19817          * @param x {int} the X position of the original mousedown
19818          * @param y {int} the Y position of the original mousedown
19819          * @static
19820          */
19821         startDrag: function(x, y) {
19822             clearTimeout(this.clickTimeout);
19823             if (this.dragCurrent) {
19824                 this.dragCurrent.b4StartDrag(x, y);
19825                 this.dragCurrent.startDrag(x, y);
19826             }
19827             this.dragThreshMet = true;
19828         },
19829
19830         /**
19831          * Internal function to handle the mouseup event.  Will be invoked
19832          * from the context of the document.
19833          * @method handleMouseUp
19834          * @param {Event} e the event
19835          * @private
19836          * @static
19837          */
19838         handleMouseUp: function(e) {
19839
19840             if(Roo.QuickTips){
19841                 Roo.QuickTips.enable();
19842             }
19843             if (! this.dragCurrent) {
19844                 return;
19845             }
19846
19847             clearTimeout(this.clickTimeout);
19848
19849             if (this.dragThreshMet) {
19850                 this.fireEvents(e, true);
19851             } else {
19852             }
19853
19854             this.stopDrag(e);
19855
19856             this.stopEvent(e);
19857         },
19858
19859         /**
19860          * Utility to stop event propagation and event default, if these
19861          * features are turned on.
19862          * @method stopEvent
19863          * @param {Event} e the event as returned by this.getEvent()
19864          * @static
19865          */
19866         stopEvent: function(e){
19867             if(this.stopPropagation) {
19868                 e.stopPropagation();
19869             }
19870
19871             if (this.preventDefault) {
19872                 e.preventDefault();
19873             }
19874         },
19875
19876         /**
19877          * Internal function to clean up event handlers after the drag
19878          * operation is complete
19879          * @method stopDrag
19880          * @param {Event} e the event
19881          * @private
19882          * @static
19883          */
19884         stopDrag: function(e) {
19885             // Fire the drag end event for the item that was dragged
19886             if (this.dragCurrent) {
19887                 if (this.dragThreshMet) {
19888                     this.dragCurrent.b4EndDrag(e);
19889                     this.dragCurrent.endDrag(e);
19890                 }
19891
19892                 this.dragCurrent.onMouseUp(e);
19893             }
19894
19895             this.dragCurrent = null;
19896             this.dragOvers = {};
19897         },
19898
19899         /**
19900          * Internal function to handle the mousemove event.  Will be invoked
19901          * from the context of the html element.
19902          *
19903          * @TODO figure out what we can do about mouse events lost when the
19904          * user drags objects beyond the window boundary.  Currently we can
19905          * detect this in internet explorer by verifying that the mouse is
19906          * down during the mousemove event.  Firefox doesn't give us the
19907          * button state on the mousemove event.
19908          * @method handleMouseMove
19909          * @param {Event} e the event
19910          * @private
19911          * @static
19912          */
19913         handleMouseMove: function(e) {
19914             if (! this.dragCurrent) {
19915                 return true;
19916             }
19917
19918             // var button = e.which || e.button;
19919
19920             // check for IE mouseup outside of page boundary
19921             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19922                 this.stopEvent(e);
19923                 return this.handleMouseUp(e);
19924             }
19925
19926             if (!this.dragThreshMet) {
19927                 var diffX = Math.abs(this.startX - e.getPageX());
19928                 var diffY = Math.abs(this.startY - e.getPageY());
19929                 if (diffX > this.clickPixelThresh ||
19930                             diffY > this.clickPixelThresh) {
19931                     this.startDrag(this.startX, this.startY);
19932                 }
19933             }
19934
19935             if (this.dragThreshMet) {
19936                 this.dragCurrent.b4Drag(e);
19937                 this.dragCurrent.onDrag(e);
19938                 if(!this.dragCurrent.moveOnly){
19939                     this.fireEvents(e, false);
19940                 }
19941             }
19942
19943             this.stopEvent(e);
19944
19945             return true;
19946         },
19947
19948         /**
19949          * Iterates over all of the DragDrop elements to find ones we are
19950          * hovering over or dropping on
19951          * @method fireEvents
19952          * @param {Event} e the event
19953          * @param {boolean} isDrop is this a drop op or a mouseover op?
19954          * @private
19955          * @static
19956          */
19957         fireEvents: function(e, isDrop) {
19958             var dc = this.dragCurrent;
19959
19960             // If the user did the mouse up outside of the window, we could
19961             // get here even though we have ended the drag.
19962             if (!dc || dc.isLocked()) {
19963                 return;
19964             }
19965
19966             var pt = e.getPoint();
19967
19968             // cache the previous dragOver array
19969             var oldOvers = [];
19970
19971             var outEvts   = [];
19972             var overEvts  = [];
19973             var dropEvts  = [];
19974             var enterEvts = [];
19975
19976             // Check to see if the object(s) we were hovering over is no longer
19977             // being hovered over so we can fire the onDragOut event
19978             for (var i in this.dragOvers) {
19979
19980                 var ddo = this.dragOvers[i];
19981
19982                 if (! this.isTypeOfDD(ddo)) {
19983                     continue;
19984                 }
19985
19986                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19987                     outEvts.push( ddo );
19988                 }
19989
19990                 oldOvers[i] = true;
19991                 delete this.dragOvers[i];
19992             }
19993
19994             for (var sGroup in dc.groups) {
19995
19996                 if ("string" != typeof sGroup) {
19997                     continue;
19998                 }
19999
20000                 for (i in this.ids[sGroup]) {
20001                     var oDD = this.ids[sGroup][i];
20002                     if (! this.isTypeOfDD(oDD)) {
20003                         continue;
20004                     }
20005
20006                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
20007                         if (this.isOverTarget(pt, oDD, this.mode)) {
20008                             // look for drop interactions
20009                             if (isDrop) {
20010                                 dropEvts.push( oDD );
20011                             // look for drag enter and drag over interactions
20012                             } else {
20013
20014                                 // initial drag over: dragEnter fires
20015                                 if (!oldOvers[oDD.id]) {
20016                                     enterEvts.push( oDD );
20017                                 // subsequent drag overs: dragOver fires
20018                                 } else {
20019                                     overEvts.push( oDD );
20020                                 }
20021
20022                                 this.dragOvers[oDD.id] = oDD;
20023                             }
20024                         }
20025                     }
20026                 }
20027             }
20028
20029             if (this.mode) {
20030                 if (outEvts.length) {
20031                     dc.b4DragOut(e, outEvts);
20032                     dc.onDragOut(e, outEvts);
20033                 }
20034
20035                 if (enterEvts.length) {
20036                     dc.onDragEnter(e, enterEvts);
20037                 }
20038
20039                 if (overEvts.length) {
20040                     dc.b4DragOver(e, overEvts);
20041                     dc.onDragOver(e, overEvts);
20042                 }
20043
20044                 if (dropEvts.length) {
20045                     dc.b4DragDrop(e, dropEvts);
20046                     dc.onDragDrop(e, dropEvts);
20047                 }
20048
20049             } else {
20050                 // fire dragout events
20051                 var len = 0;
20052                 for (i=0, len=outEvts.length; i<len; ++i) {
20053                     dc.b4DragOut(e, outEvts[i].id);
20054                     dc.onDragOut(e, outEvts[i].id);
20055                 }
20056
20057                 // fire enter events
20058                 for (i=0,len=enterEvts.length; i<len; ++i) {
20059                     // dc.b4DragEnter(e, oDD.id);
20060                     dc.onDragEnter(e, enterEvts[i].id);
20061                 }
20062
20063                 // fire over events
20064                 for (i=0,len=overEvts.length; i<len; ++i) {
20065                     dc.b4DragOver(e, overEvts[i].id);
20066                     dc.onDragOver(e, overEvts[i].id);
20067                 }
20068
20069                 // fire drop events
20070                 for (i=0, len=dropEvts.length; i<len; ++i) {
20071                     dc.b4DragDrop(e, dropEvts[i].id);
20072                     dc.onDragDrop(e, dropEvts[i].id);
20073                 }
20074
20075             }
20076
20077             // notify about a drop that did not find a target
20078             if (isDrop && !dropEvts.length) {
20079                 dc.onInvalidDrop(e);
20080             }
20081
20082         },
20083
20084         /**
20085          * Helper function for getting the best match from the list of drag
20086          * and drop objects returned by the drag and drop events when we are
20087          * in INTERSECT mode.  It returns either the first object that the
20088          * cursor is over, or the object that has the greatest overlap with
20089          * the dragged element.
20090          * @method getBestMatch
20091          * @param  {DragDrop[]} dds The array of drag and drop objects
20092          * targeted
20093          * @return {DragDrop}       The best single match
20094          * @static
20095          */
20096         getBestMatch: function(dds) {
20097             var winner = null;
20098             // Return null if the input is not what we expect
20099             //if (!dds || !dds.length || dds.length == 0) {
20100                // winner = null;
20101             // If there is only one item, it wins
20102             //} else if (dds.length == 1) {
20103
20104             var len = dds.length;
20105
20106             if (len == 1) {
20107                 winner = dds[0];
20108             } else {
20109                 // Loop through the targeted items
20110                 for (var i=0; i<len; ++i) {
20111                     var dd = dds[i];
20112                     // If the cursor is over the object, it wins.  If the
20113                     // cursor is over multiple matches, the first one we come
20114                     // to wins.
20115                     if (dd.cursorIsOver) {
20116                         winner = dd;
20117                         break;
20118                     // Otherwise the object with the most overlap wins
20119                     } else {
20120                         if (!winner ||
20121                             winner.overlap.getArea() < dd.overlap.getArea()) {
20122                             winner = dd;
20123                         }
20124                     }
20125                 }
20126             }
20127
20128             return winner;
20129         },
20130
20131         /**
20132          * Refreshes the cache of the top-left and bottom-right points of the
20133          * drag and drop objects in the specified group(s).  This is in the
20134          * format that is stored in the drag and drop instance, so typical
20135          * usage is:
20136          * <code>
20137          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20138          * </code>
20139          * Alternatively:
20140          * <code>
20141          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20142          * </code>
20143          * @TODO this really should be an indexed array.  Alternatively this
20144          * method could accept both.
20145          * @method refreshCache
20146          * @param {Object} groups an associative array of groups to refresh
20147          * @static
20148          */
20149         refreshCache: function(groups) {
20150             for (var sGroup in groups) {
20151                 if ("string" != typeof sGroup) {
20152                     continue;
20153                 }
20154                 for (var i in this.ids[sGroup]) {
20155                     var oDD = this.ids[sGroup][i];
20156
20157                     if (this.isTypeOfDD(oDD)) {
20158                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20159                         var loc = this.getLocation(oDD);
20160                         if (loc) {
20161                             this.locationCache[oDD.id] = loc;
20162                         } else {
20163                             delete this.locationCache[oDD.id];
20164                             // this will unregister the drag and drop object if
20165                             // the element is not in a usable state
20166                             // oDD.unreg();
20167                         }
20168                     }
20169                 }
20170             }
20171         },
20172
20173         /**
20174          * This checks to make sure an element exists and is in the DOM.  The
20175          * main purpose is to handle cases where innerHTML is used to remove
20176          * drag and drop objects from the DOM.  IE provides an 'unspecified
20177          * error' when trying to access the offsetParent of such an element
20178          * @method verifyEl
20179          * @param {HTMLElement} el the element to check
20180          * @return {boolean} true if the element looks usable
20181          * @static
20182          */
20183         verifyEl: function(el) {
20184             if (el) {
20185                 var parent;
20186                 if(Roo.isIE){
20187                     try{
20188                         parent = el.offsetParent;
20189                     }catch(e){}
20190                 }else{
20191                     parent = el.offsetParent;
20192                 }
20193                 if (parent) {
20194                     return true;
20195                 }
20196             }
20197
20198             return false;
20199         },
20200
20201         /**
20202          * Returns a Region object containing the drag and drop element's position
20203          * and size, including the padding configured for it
20204          * @method getLocation
20205          * @param {DragDrop} oDD the drag and drop object to get the
20206          *                       location for
20207          * @return {Roo.lib.Region} a Region object representing the total area
20208          *                             the element occupies, including any padding
20209          *                             the instance is configured for.
20210          * @static
20211          */
20212         getLocation: function(oDD) {
20213             if (! this.isTypeOfDD(oDD)) {
20214                 return null;
20215             }
20216
20217             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20218
20219             try {
20220                 pos= Roo.lib.Dom.getXY(el);
20221             } catch (e) { }
20222
20223             if (!pos) {
20224                 return null;
20225             }
20226
20227             x1 = pos[0];
20228             x2 = x1 + el.offsetWidth;
20229             y1 = pos[1];
20230             y2 = y1 + el.offsetHeight;
20231
20232             t = y1 - oDD.padding[0];
20233             r = x2 + oDD.padding[1];
20234             b = y2 + oDD.padding[2];
20235             l = x1 - oDD.padding[3];
20236
20237             return new Roo.lib.Region( t, r, b, l );
20238         },
20239
20240         /**
20241          * Checks the cursor location to see if it over the target
20242          * @method isOverTarget
20243          * @param {Roo.lib.Point} pt The point to evaluate
20244          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20245          * @return {boolean} true if the mouse is over the target
20246          * @private
20247          * @static
20248          */
20249         isOverTarget: function(pt, oTarget, intersect) {
20250             // use cache if available
20251             var loc = this.locationCache[oTarget.id];
20252             if (!loc || !this.useCache) {
20253                 loc = this.getLocation(oTarget);
20254                 this.locationCache[oTarget.id] = loc;
20255
20256             }
20257
20258             if (!loc) {
20259                 return false;
20260             }
20261
20262             oTarget.cursorIsOver = loc.contains( pt );
20263
20264             // DragDrop is using this as a sanity check for the initial mousedown
20265             // in this case we are done.  In POINT mode, if the drag obj has no
20266             // contraints, we are also done. Otherwise we need to evaluate the
20267             // location of the target as related to the actual location of the
20268             // dragged element.
20269             var dc = this.dragCurrent;
20270             if (!dc || !dc.getTargetCoord ||
20271                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20272                 return oTarget.cursorIsOver;
20273             }
20274
20275             oTarget.overlap = null;
20276
20277             // Get the current location of the drag element, this is the
20278             // location of the mouse event less the delta that represents
20279             // where the original mousedown happened on the element.  We
20280             // need to consider constraints and ticks as well.
20281             var pos = dc.getTargetCoord(pt.x, pt.y);
20282
20283             var el = dc.getDragEl();
20284             var curRegion = new Roo.lib.Region( pos.y,
20285                                                    pos.x + el.offsetWidth,
20286                                                    pos.y + el.offsetHeight,
20287                                                    pos.x );
20288
20289             var overlap = curRegion.intersect(loc);
20290
20291             if (overlap) {
20292                 oTarget.overlap = overlap;
20293                 return (intersect) ? true : oTarget.cursorIsOver;
20294             } else {
20295                 return false;
20296             }
20297         },
20298
20299         /**
20300          * unload event handler
20301          * @method _onUnload
20302          * @private
20303          * @static
20304          */
20305         _onUnload: function(e, me) {
20306             Roo.dd.DragDropMgr.unregAll();
20307         },
20308
20309         /**
20310          * Cleans up the drag and drop events and objects.
20311          * @method unregAll
20312          * @private
20313          * @static
20314          */
20315         unregAll: function() {
20316
20317             if (this.dragCurrent) {
20318                 this.stopDrag();
20319                 this.dragCurrent = null;
20320             }
20321
20322             this._execOnAll("unreg", []);
20323
20324             for (i in this.elementCache) {
20325                 delete this.elementCache[i];
20326             }
20327
20328             this.elementCache = {};
20329             this.ids = {};
20330         },
20331
20332         /**
20333          * A cache of DOM elements
20334          * @property elementCache
20335          * @private
20336          * @static
20337          */
20338         elementCache: {},
20339
20340         /**
20341          * Get the wrapper for the DOM element specified
20342          * @method getElWrapper
20343          * @param {String} id the id of the element to get
20344          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20345          * @private
20346          * @deprecated This wrapper isn't that useful
20347          * @static
20348          */
20349         getElWrapper: function(id) {
20350             var oWrapper = this.elementCache[id];
20351             if (!oWrapper || !oWrapper.el) {
20352                 oWrapper = this.elementCache[id] =
20353                     new this.ElementWrapper(Roo.getDom(id));
20354             }
20355             return oWrapper;
20356         },
20357
20358         /**
20359          * Returns the actual DOM element
20360          * @method getElement
20361          * @param {String} id the id of the elment to get
20362          * @return {Object} The element
20363          * @deprecated use Roo.getDom instead
20364          * @static
20365          */
20366         getElement: function(id) {
20367             return Roo.getDom(id);
20368         },
20369
20370         /**
20371          * Returns the style property for the DOM element (i.e.,
20372          * document.getElById(id).style)
20373          * @method getCss
20374          * @param {String} id the id of the elment to get
20375          * @return {Object} The style property of the element
20376          * @deprecated use Roo.getDom instead
20377          * @static
20378          */
20379         getCss: function(id) {
20380             var el = Roo.getDom(id);
20381             return (el) ? el.style : null;
20382         },
20383
20384         /**
20385          * Inner class for cached elements
20386          * @class DragDropMgr.ElementWrapper
20387          * @for DragDropMgr
20388          * @private
20389          * @deprecated
20390          */
20391         ElementWrapper: function(el) {
20392                 /**
20393                  * The element
20394                  * @property el
20395                  */
20396                 this.el = el || null;
20397                 /**
20398                  * The element id
20399                  * @property id
20400                  */
20401                 this.id = this.el && el.id;
20402                 /**
20403                  * A reference to the style property
20404                  * @property css
20405                  */
20406                 this.css = this.el && el.style;
20407             },
20408
20409         /**
20410          * Returns the X position of an html element
20411          * @method getPosX
20412          * @param el the element for which to get the position
20413          * @return {int} the X coordinate
20414          * @for DragDropMgr
20415          * @deprecated use Roo.lib.Dom.getX instead
20416          * @static
20417          */
20418         getPosX: function(el) {
20419             return Roo.lib.Dom.getX(el);
20420         },
20421
20422         /**
20423          * Returns the Y position of an html element
20424          * @method getPosY
20425          * @param el the element for which to get the position
20426          * @return {int} the Y coordinate
20427          * @deprecated use Roo.lib.Dom.getY instead
20428          * @static
20429          */
20430         getPosY: function(el) {
20431             return Roo.lib.Dom.getY(el);
20432         },
20433
20434         /**
20435          * Swap two nodes.  In IE, we use the native method, for others we
20436          * emulate the IE behavior
20437          * @method swapNode
20438          * @param n1 the first node to swap
20439          * @param n2 the other node to swap
20440          * @static
20441          */
20442         swapNode: function(n1, n2) {
20443             if (n1.swapNode) {
20444                 n1.swapNode(n2);
20445             } else {
20446                 var p = n2.parentNode;
20447                 var s = n2.nextSibling;
20448
20449                 if (s == n1) {
20450                     p.insertBefore(n1, n2);
20451                 } else if (n2 == n1.nextSibling) {
20452                     p.insertBefore(n2, n1);
20453                 } else {
20454                     n1.parentNode.replaceChild(n2, n1);
20455                     p.insertBefore(n1, s);
20456                 }
20457             }
20458         },
20459
20460         /**
20461          * Returns the current scroll position
20462          * @method getScroll
20463          * @private
20464          * @static
20465          */
20466         getScroll: function () {
20467             var t, l, dde=document.documentElement, db=document.body;
20468             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20469                 t = dde.scrollTop;
20470                 l = dde.scrollLeft;
20471             } else if (db) {
20472                 t = db.scrollTop;
20473                 l = db.scrollLeft;
20474             } else {
20475
20476             }
20477             return { top: t, left: l };
20478         },
20479
20480         /**
20481          * Returns the specified element style property
20482          * @method getStyle
20483          * @param {HTMLElement} el          the element
20484          * @param {string}      styleProp   the style property
20485          * @return {string} The value of the style property
20486          * @deprecated use Roo.lib.Dom.getStyle
20487          * @static
20488          */
20489         getStyle: function(el, styleProp) {
20490             return Roo.fly(el).getStyle(styleProp);
20491         },
20492
20493         /**
20494          * Gets the scrollTop
20495          * @method getScrollTop
20496          * @return {int} the document's scrollTop
20497          * @static
20498          */
20499         getScrollTop: function () { return this.getScroll().top; },
20500
20501         /**
20502          * Gets the scrollLeft
20503          * @method getScrollLeft
20504          * @return {int} the document's scrollTop
20505          * @static
20506          */
20507         getScrollLeft: function () { return this.getScroll().left; },
20508
20509         /**
20510          * Sets the x/y position of an element to the location of the
20511          * target element.
20512          * @method moveToEl
20513          * @param {HTMLElement} moveEl      The element to move
20514          * @param {HTMLElement} targetEl    The position reference element
20515          * @static
20516          */
20517         moveToEl: function (moveEl, targetEl) {
20518             var aCoord = Roo.lib.Dom.getXY(targetEl);
20519             Roo.lib.Dom.setXY(moveEl, aCoord);
20520         },
20521
20522         /**
20523          * Numeric array sort function
20524          * @method numericSort
20525          * @static
20526          */
20527         numericSort: function(a, b) { return (a - b); },
20528
20529         /**
20530          * Internal counter
20531          * @property _timeoutCount
20532          * @private
20533          * @static
20534          */
20535         _timeoutCount: 0,
20536
20537         /**
20538          * Trying to make the load order less important.  Without this we get
20539          * an error if this file is loaded before the Event Utility.
20540          * @method _addListeners
20541          * @private
20542          * @static
20543          */
20544         _addListeners: function() {
20545             var DDM = Roo.dd.DDM;
20546             if ( Roo.lib.Event && document ) {
20547                 DDM._onLoad();
20548             } else {
20549                 if (DDM._timeoutCount > 2000) {
20550                 } else {
20551                     setTimeout(DDM._addListeners, 10);
20552                     if (document && document.body) {
20553                         DDM._timeoutCount += 1;
20554                     }
20555                 }
20556             }
20557         },
20558
20559         /**
20560          * Recursively searches the immediate parent and all child nodes for
20561          * the handle element in order to determine wheter or not it was
20562          * clicked.
20563          * @method handleWasClicked
20564          * @param node the html element to inspect
20565          * @static
20566          */
20567         handleWasClicked: function(node, id) {
20568             if (this.isHandle(id, node.id)) {
20569                 return true;
20570             } else {
20571                 // check to see if this is a text node child of the one we want
20572                 var p = node.parentNode;
20573
20574                 while (p) {
20575                     if (this.isHandle(id, p.id)) {
20576                         return true;
20577                     } else {
20578                         p = p.parentNode;
20579                     }
20580                 }
20581             }
20582
20583             return false;
20584         }
20585
20586     };
20587
20588 }();
20589
20590 // shorter alias, save a few bytes
20591 Roo.dd.DDM = Roo.dd.DragDropMgr;
20592 Roo.dd.DDM._addListeners();
20593
20594 }/*
20595  * Based on:
20596  * Ext JS Library 1.1.1
20597  * Copyright(c) 2006-2007, Ext JS, LLC.
20598  *
20599  * Originally Released Under LGPL - original licence link has changed is not relivant.
20600  *
20601  * Fork - LGPL
20602  * <script type="text/javascript">
20603  */
20604
20605 /**
20606  * @class Roo.dd.DD
20607  * A DragDrop implementation where the linked element follows the
20608  * mouse cursor during a drag.
20609  * @extends Roo.dd.DragDrop
20610  * @constructor
20611  * @param {String} id the id of the linked element
20612  * @param {String} sGroup the group of related DragDrop items
20613  * @param {object} config an object containing configurable attributes
20614  *                Valid properties for DD:
20615  *                    scroll
20616  */
20617 Roo.dd.DD = function(id, sGroup, config) {
20618     if (id) {
20619         this.init(id, sGroup, config);
20620     }
20621 };
20622
20623 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20624
20625     /**
20626      * When set to true, the utility automatically tries to scroll the browser
20627      * window wehn a drag and drop element is dragged near the viewport boundary.
20628      * Defaults to true.
20629      * @property scroll
20630      * @type boolean
20631      */
20632     scroll: true,
20633
20634     /**
20635      * Sets the pointer offset to the distance between the linked element's top
20636      * left corner and the location the element was clicked
20637      * @method autoOffset
20638      * @param {int} iPageX the X coordinate of the click
20639      * @param {int} iPageY the Y coordinate of the click
20640      */
20641     autoOffset: function(iPageX, iPageY) {
20642         var x = iPageX - this.startPageX;
20643         var y = iPageY - this.startPageY;
20644         this.setDelta(x, y);
20645     },
20646
20647     /**
20648      * Sets the pointer offset.  You can call this directly to force the
20649      * offset to be in a particular location (e.g., pass in 0,0 to set it
20650      * to the center of the object)
20651      * @method setDelta
20652      * @param {int} iDeltaX the distance from the left
20653      * @param {int} iDeltaY the distance from the top
20654      */
20655     setDelta: function(iDeltaX, iDeltaY) {
20656         this.deltaX = iDeltaX;
20657         this.deltaY = iDeltaY;
20658     },
20659
20660     /**
20661      * Sets the drag element to the location of the mousedown or click event,
20662      * maintaining the cursor location relative to the location on the element
20663      * that was clicked.  Override this if you want to place the element in a
20664      * location other than where the cursor is.
20665      * @method setDragElPos
20666      * @param {int} iPageX the X coordinate of the mousedown or drag event
20667      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20668      */
20669     setDragElPos: function(iPageX, iPageY) {
20670         // the first time we do this, we are going to check to make sure
20671         // the element has css positioning
20672
20673         var el = this.getDragEl();
20674         this.alignElWithMouse(el, iPageX, iPageY);
20675     },
20676
20677     /**
20678      * Sets the element to the location of the mousedown or click event,
20679      * maintaining the cursor location relative to the location on the element
20680      * that was clicked.  Override this if you want to place the element in a
20681      * location other than where the cursor is.
20682      * @method alignElWithMouse
20683      * @param {HTMLElement} el the element to move
20684      * @param {int} iPageX the X coordinate of the mousedown or drag event
20685      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20686      */
20687     alignElWithMouse: function(el, iPageX, iPageY) {
20688         var oCoord = this.getTargetCoord(iPageX, iPageY);
20689         var fly = el.dom ? el : Roo.fly(el);
20690         if (!this.deltaSetXY) {
20691             var aCoord = [oCoord.x, oCoord.y];
20692             fly.setXY(aCoord);
20693             var newLeft = fly.getLeft(true);
20694             var newTop  = fly.getTop(true);
20695             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20696         } else {
20697             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20698         }
20699
20700         this.cachePosition(oCoord.x, oCoord.y);
20701         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20702         return oCoord;
20703     },
20704
20705     /**
20706      * Saves the most recent position so that we can reset the constraints and
20707      * tick marks on-demand.  We need to know this so that we can calculate the
20708      * number of pixels the element is offset from its original position.
20709      * @method cachePosition
20710      * @param iPageX the current x position (optional, this just makes it so we
20711      * don't have to look it up again)
20712      * @param iPageY the current y position (optional, this just makes it so we
20713      * don't have to look it up again)
20714      */
20715     cachePosition: function(iPageX, iPageY) {
20716         if (iPageX) {
20717             this.lastPageX = iPageX;
20718             this.lastPageY = iPageY;
20719         } else {
20720             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20721             this.lastPageX = aCoord[0];
20722             this.lastPageY = aCoord[1];
20723         }
20724     },
20725
20726     /**
20727      * Auto-scroll the window if the dragged object has been moved beyond the
20728      * visible window boundary.
20729      * @method autoScroll
20730      * @param {int} x the drag element's x position
20731      * @param {int} y the drag element's y position
20732      * @param {int} h the height of the drag element
20733      * @param {int} w the width of the drag element
20734      * @private
20735      */
20736     autoScroll: function(x, y, h, w) {
20737
20738         if (this.scroll) {
20739             // The client height
20740             var clientH = Roo.lib.Dom.getViewWidth();
20741
20742             // The client width
20743             var clientW = Roo.lib.Dom.getViewHeight();
20744
20745             // The amt scrolled down
20746             var st = this.DDM.getScrollTop();
20747
20748             // The amt scrolled right
20749             var sl = this.DDM.getScrollLeft();
20750
20751             // Location of the bottom of the element
20752             var bot = h + y;
20753
20754             // Location of the right of the element
20755             var right = w + x;
20756
20757             // The distance from the cursor to the bottom of the visible area,
20758             // adjusted so that we don't scroll if the cursor is beyond the
20759             // element drag constraints
20760             var toBot = (clientH + st - y - this.deltaY);
20761
20762             // The distance from the cursor to the right of the visible area
20763             var toRight = (clientW + sl - x - this.deltaX);
20764
20765
20766             // How close to the edge the cursor must be before we scroll
20767             // var thresh = (document.all) ? 100 : 40;
20768             var thresh = 40;
20769
20770             // How many pixels to scroll per autoscroll op.  This helps to reduce
20771             // clunky scrolling. IE is more sensitive about this ... it needs this
20772             // value to be higher.
20773             var scrAmt = (document.all) ? 80 : 30;
20774
20775             // Scroll down if we are near the bottom of the visible page and the
20776             // obj extends below the crease
20777             if ( bot > clientH && toBot < thresh ) {
20778                 window.scrollTo(sl, st + scrAmt);
20779             }
20780
20781             // Scroll up if the window is scrolled down and the top of the object
20782             // goes above the top border
20783             if ( y < st && st > 0 && y - st < thresh ) {
20784                 window.scrollTo(sl, st - scrAmt);
20785             }
20786
20787             // Scroll right if the obj is beyond the right border and the cursor is
20788             // near the border.
20789             if ( right > clientW && toRight < thresh ) {
20790                 window.scrollTo(sl + scrAmt, st);
20791             }
20792
20793             // Scroll left if the window has been scrolled to the right and the obj
20794             // extends past the left border
20795             if ( x < sl && sl > 0 && x - sl < thresh ) {
20796                 window.scrollTo(sl - scrAmt, st);
20797             }
20798         }
20799     },
20800
20801     /**
20802      * Finds the location the element should be placed if we want to move
20803      * it to where the mouse location less the click offset would place us.
20804      * @method getTargetCoord
20805      * @param {int} iPageX the X coordinate of the click
20806      * @param {int} iPageY the Y coordinate of the click
20807      * @return an object that contains the coordinates (Object.x and Object.y)
20808      * @private
20809      */
20810     getTargetCoord: function(iPageX, iPageY) {
20811
20812
20813         var x = iPageX - this.deltaX;
20814         var y = iPageY - this.deltaY;
20815
20816         if (this.constrainX) {
20817             if (x < this.minX) { x = this.minX; }
20818             if (x > this.maxX) { x = this.maxX; }
20819         }
20820
20821         if (this.constrainY) {
20822             if (y < this.minY) { y = this.minY; }
20823             if (y > this.maxY) { y = this.maxY; }
20824         }
20825
20826         x = this.getTick(x, this.xTicks);
20827         y = this.getTick(y, this.yTicks);
20828
20829
20830         return {x:x, y:y};
20831     },
20832
20833     /*
20834      * Sets up config options specific to this class. Overrides
20835      * Roo.dd.DragDrop, but all versions of this method through the
20836      * inheritance chain are called
20837      */
20838     applyConfig: function() {
20839         Roo.dd.DD.superclass.applyConfig.call(this);
20840         this.scroll = (this.config.scroll !== false);
20841     },
20842
20843     /*
20844      * Event that fires prior to the onMouseDown event.  Overrides
20845      * Roo.dd.DragDrop.
20846      */
20847     b4MouseDown: function(e) {
20848         // this.resetConstraints();
20849         this.autoOffset(e.getPageX(),
20850                             e.getPageY());
20851     },
20852
20853     /*
20854      * Event that fires prior to the onDrag event.  Overrides
20855      * Roo.dd.DragDrop.
20856      */
20857     b4Drag: function(e) {
20858         this.setDragElPos(e.getPageX(),
20859                             e.getPageY());
20860     },
20861
20862     toString: function() {
20863         return ("DD " + this.id);
20864     }
20865
20866     //////////////////////////////////////////////////////////////////////////
20867     // Debugging ygDragDrop events that can be overridden
20868     //////////////////////////////////////////////////////////////////////////
20869     /*
20870     startDrag: function(x, y) {
20871     },
20872
20873     onDrag: function(e) {
20874     },
20875
20876     onDragEnter: function(e, id) {
20877     },
20878
20879     onDragOver: function(e, id) {
20880     },
20881
20882     onDragOut: function(e, id) {
20883     },
20884
20885     onDragDrop: function(e, id) {
20886     },
20887
20888     endDrag: function(e) {
20889     }
20890
20891     */
20892
20893 });/*
20894  * Based on:
20895  * Ext JS Library 1.1.1
20896  * Copyright(c) 2006-2007, Ext JS, LLC.
20897  *
20898  * Originally Released Under LGPL - original licence link has changed is not relivant.
20899  *
20900  * Fork - LGPL
20901  * <script type="text/javascript">
20902  */
20903
20904 /**
20905  * @class Roo.dd.DDProxy
20906  * A DragDrop implementation that inserts an empty, bordered div into
20907  * the document that follows the cursor during drag operations.  At the time of
20908  * the click, the frame div is resized to the dimensions of the linked html
20909  * element, and moved to the exact location of the linked element.
20910  *
20911  * References to the "frame" element refer to the single proxy element that
20912  * was created to be dragged in place of all DDProxy elements on the
20913  * page.
20914  *
20915  * @extends Roo.dd.DD
20916  * @constructor
20917  * @param {String} id the id of the linked html element
20918  * @param {String} sGroup the group of related DragDrop objects
20919  * @param {object} config an object containing configurable attributes
20920  *                Valid properties for DDProxy in addition to those in DragDrop:
20921  *                   resizeFrame, centerFrame, dragElId
20922  */
20923 Roo.dd.DDProxy = function(id, sGroup, config) {
20924     if (id) {
20925         this.init(id, sGroup, config);
20926         this.initFrame();
20927     }
20928 };
20929
20930 /**
20931  * The default drag frame div id
20932  * @property Roo.dd.DDProxy.dragElId
20933  * @type String
20934  * @static
20935  */
20936 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20937
20938 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20939
20940     /**
20941      * By default we resize the drag frame to be the same size as the element
20942      * we want to drag (this is to get the frame effect).  We can turn it off
20943      * if we want a different behavior.
20944      * @property resizeFrame
20945      * @type boolean
20946      */
20947     resizeFrame: true,
20948
20949     /**
20950      * By default the frame is positioned exactly where the drag element is, so
20951      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20952      * you do not have constraints on the obj is to have the drag frame centered
20953      * around the cursor.  Set centerFrame to true for this effect.
20954      * @property centerFrame
20955      * @type boolean
20956      */
20957     centerFrame: false,
20958
20959     /**
20960      * Creates the proxy element if it does not yet exist
20961      * @method createFrame
20962      */
20963     createFrame: function() {
20964         var self = this;
20965         var body = document.body;
20966
20967         if (!body || !body.firstChild) {
20968             setTimeout( function() { self.createFrame(); }, 50 );
20969             return;
20970         }
20971
20972         var div = this.getDragEl();
20973
20974         if (!div) {
20975             div    = document.createElement("div");
20976             div.id = this.dragElId;
20977             var s  = div.style;
20978
20979             s.position   = "absolute";
20980             s.visibility = "hidden";
20981             s.cursor     = "move";
20982             s.border     = "2px solid #aaa";
20983             s.zIndex     = 999;
20984
20985             // appendChild can blow up IE if invoked prior to the window load event
20986             // while rendering a table.  It is possible there are other scenarios
20987             // that would cause this to happen as well.
20988             body.insertBefore(div, body.firstChild);
20989         }
20990     },
20991
20992     /**
20993      * Initialization for the drag frame element.  Must be called in the
20994      * constructor of all subclasses
20995      * @method initFrame
20996      */
20997     initFrame: function() {
20998         this.createFrame();
20999     },
21000
21001     applyConfig: function() {
21002         Roo.dd.DDProxy.superclass.applyConfig.call(this);
21003
21004         this.resizeFrame = (this.config.resizeFrame !== false);
21005         this.centerFrame = (this.config.centerFrame);
21006         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
21007     },
21008
21009     /**
21010      * Resizes the drag frame to the dimensions of the clicked object, positions
21011      * it over the object, and finally displays it
21012      * @method showFrame
21013      * @param {int} iPageX X click position
21014      * @param {int} iPageY Y click position
21015      * @private
21016      */
21017     showFrame: function(iPageX, iPageY) {
21018         var el = this.getEl();
21019         var dragEl = this.getDragEl();
21020         var s = dragEl.style;
21021
21022         this._resizeProxy();
21023
21024         if (this.centerFrame) {
21025             this.setDelta( Math.round(parseInt(s.width,  10)/2),
21026                            Math.round(parseInt(s.height, 10)/2) );
21027         }
21028
21029         this.setDragElPos(iPageX, iPageY);
21030
21031         Roo.fly(dragEl).show();
21032     },
21033
21034     /**
21035      * The proxy is automatically resized to the dimensions of the linked
21036      * element when a drag is initiated, unless resizeFrame is set to false
21037      * @method _resizeProxy
21038      * @private
21039      */
21040     _resizeProxy: function() {
21041         if (this.resizeFrame) {
21042             var el = this.getEl();
21043             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21044         }
21045     },
21046
21047     // overrides Roo.dd.DragDrop
21048     b4MouseDown: function(e) {
21049         var x = e.getPageX();
21050         var y = e.getPageY();
21051         this.autoOffset(x, y);
21052         this.setDragElPos(x, y);
21053     },
21054
21055     // overrides Roo.dd.DragDrop
21056     b4StartDrag: function(x, y) {
21057         // show the drag frame
21058         this.showFrame(x, y);
21059     },
21060
21061     // overrides Roo.dd.DragDrop
21062     b4EndDrag: function(e) {
21063         Roo.fly(this.getDragEl()).hide();
21064     },
21065
21066     // overrides Roo.dd.DragDrop
21067     // By default we try to move the element to the last location of the frame.
21068     // This is so that the default behavior mirrors that of Roo.dd.DD.
21069     endDrag: function(e) {
21070
21071         var lel = this.getEl();
21072         var del = this.getDragEl();
21073
21074         // Show the drag frame briefly so we can get its position
21075         del.style.visibility = "";
21076
21077         this.beforeMove();
21078         // Hide the linked element before the move to get around a Safari
21079         // rendering bug.
21080         lel.style.visibility = "hidden";
21081         Roo.dd.DDM.moveToEl(lel, del);
21082         del.style.visibility = "hidden";
21083         lel.style.visibility = "";
21084
21085         this.afterDrag();
21086     },
21087
21088     beforeMove : function(){
21089
21090     },
21091
21092     afterDrag : function(){
21093
21094     },
21095
21096     toString: function() {
21097         return ("DDProxy " + this.id);
21098     }
21099
21100 });
21101 /*
21102  * Based on:
21103  * Ext JS Library 1.1.1
21104  * Copyright(c) 2006-2007, Ext JS, LLC.
21105  *
21106  * Originally Released Under LGPL - original licence link has changed is not relivant.
21107  *
21108  * Fork - LGPL
21109  * <script type="text/javascript">
21110  */
21111
21112  /**
21113  * @class Roo.dd.DDTarget
21114  * A DragDrop implementation that does not move, but can be a drop
21115  * target.  You would get the same result by simply omitting implementation
21116  * for the event callbacks, but this way we reduce the processing cost of the
21117  * event listener and the callbacks.
21118  * @extends Roo.dd.DragDrop
21119  * @constructor
21120  * @param {String} id the id of the element that is a drop target
21121  * @param {String} sGroup the group of related DragDrop objects
21122  * @param {object} config an object containing configurable attributes
21123  *                 Valid properties for DDTarget in addition to those in
21124  *                 DragDrop:
21125  *                    none
21126  */
21127 Roo.dd.DDTarget = function(id, sGroup, config) {
21128     if (id) {
21129         this.initTarget(id, sGroup, config);
21130     }
21131     if (config && (config.listeners || config.events)) { 
21132         Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21133             listeners : config.listeners || {}, 
21134             events : config.events || {} 
21135         });    
21136     }
21137 };
21138
21139 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21140 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21141     toString: function() {
21142         return ("DDTarget " + this.id);
21143     }
21144 });
21145 /*
21146  * Based on:
21147  * Ext JS Library 1.1.1
21148  * Copyright(c) 2006-2007, Ext JS, LLC.
21149  *
21150  * Originally Released Under LGPL - original licence link has changed is not relivant.
21151  *
21152  * Fork - LGPL
21153  * <script type="text/javascript">
21154  */
21155  
21156
21157 /**
21158  * @class Roo.dd.ScrollManager
21159  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21160  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21161  * @singleton
21162  */
21163 Roo.dd.ScrollManager = function(){
21164     var ddm = Roo.dd.DragDropMgr;
21165     var els = {};
21166     var dragEl = null;
21167     var proc = {};
21168     
21169     
21170     
21171     var onStop = function(e){
21172         dragEl = null;
21173         clearProc();
21174     };
21175     
21176     var triggerRefresh = function(){
21177         if(ddm.dragCurrent){
21178              ddm.refreshCache(ddm.dragCurrent.groups);
21179         }
21180     };
21181     
21182     var doScroll = function(){
21183         if(ddm.dragCurrent){
21184             var dds = Roo.dd.ScrollManager;
21185             if(!dds.animate){
21186                 if(proc.el.scroll(proc.dir, dds.increment)){
21187                     triggerRefresh();
21188                 }
21189             }else{
21190                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21191             }
21192         }
21193     };
21194     
21195     var clearProc = function(){
21196         if(proc.id){
21197             clearInterval(proc.id);
21198         }
21199         proc.id = 0;
21200         proc.el = null;
21201         proc.dir = "";
21202     };
21203     
21204     var startProc = function(el, dir){
21205          Roo.log('scroll startproc');
21206         clearProc();
21207         proc.el = el;
21208         proc.dir = dir;
21209         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21210     };
21211     
21212     var onFire = function(e, isDrop){
21213        
21214         if(isDrop || !ddm.dragCurrent){ return; }
21215         var dds = Roo.dd.ScrollManager;
21216         if(!dragEl || dragEl != ddm.dragCurrent){
21217             dragEl = ddm.dragCurrent;
21218             // refresh regions on drag start
21219             dds.refreshCache();
21220         }
21221         
21222         var xy = Roo.lib.Event.getXY(e);
21223         var pt = new Roo.lib.Point(xy[0], xy[1]);
21224         for(var id in els){
21225             var el = els[id], r = el._region;
21226             if(r && r.contains(pt) && el.isScrollable()){
21227                 if(r.bottom - pt.y <= dds.thresh){
21228                     if(proc.el != el){
21229                         startProc(el, "down");
21230                     }
21231                     return;
21232                 }else if(r.right - pt.x <= dds.thresh){
21233                     if(proc.el != el){
21234                         startProc(el, "left");
21235                     }
21236                     return;
21237                 }else if(pt.y - r.top <= dds.thresh){
21238                     if(proc.el != el){
21239                         startProc(el, "up");
21240                     }
21241                     return;
21242                 }else if(pt.x - r.left <= dds.thresh){
21243                     if(proc.el != el){
21244                         startProc(el, "right");
21245                     }
21246                     return;
21247                 }
21248             }
21249         }
21250         clearProc();
21251     };
21252     
21253     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21254     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21255     
21256     return {
21257         /**
21258          * Registers new overflow element(s) to auto scroll
21259          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21260          */
21261         register : function(el){
21262             if(el instanceof Array){
21263                 for(var i = 0, len = el.length; i < len; i++) {
21264                         this.register(el[i]);
21265                 }
21266             }else{
21267                 el = Roo.get(el);
21268                 els[el.id] = el;
21269             }
21270             Roo.dd.ScrollManager.els = els;
21271         },
21272         
21273         /**
21274          * Unregisters overflow element(s) so they are no longer scrolled
21275          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21276          */
21277         unregister : function(el){
21278             if(el instanceof Array){
21279                 for(var i = 0, len = el.length; i < len; i++) {
21280                         this.unregister(el[i]);
21281                 }
21282             }else{
21283                 el = Roo.get(el);
21284                 delete els[el.id];
21285             }
21286         },
21287         
21288         /**
21289          * The number of pixels from the edge of a container the pointer needs to be to 
21290          * trigger scrolling (defaults to 25)
21291          * @type Number
21292          */
21293         thresh : 25,
21294         
21295         /**
21296          * The number of pixels to scroll in each scroll increment (defaults to 50)
21297          * @type Number
21298          */
21299         increment : 100,
21300         
21301         /**
21302          * The frequency of scrolls in milliseconds (defaults to 500)
21303          * @type Number
21304          */
21305         frequency : 500,
21306         
21307         /**
21308          * True to animate the scroll (defaults to true)
21309          * @type Boolean
21310          */
21311         animate: true,
21312         
21313         /**
21314          * The animation duration in seconds - 
21315          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21316          * @type Number
21317          */
21318         animDuration: .4,
21319         
21320         /**
21321          * Manually trigger a cache refresh.
21322          */
21323         refreshCache : function(){
21324             for(var id in els){
21325                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21326                     els[id]._region = els[id].getRegion();
21327                 }
21328             }
21329         }
21330     };
21331 }();/*
21332  * Based on:
21333  * Ext JS Library 1.1.1
21334  * Copyright(c) 2006-2007, Ext JS, LLC.
21335  *
21336  * Originally Released Under LGPL - original licence link has changed is not relivant.
21337  *
21338  * Fork - LGPL
21339  * <script type="text/javascript">
21340  */
21341  
21342
21343 /**
21344  * @class Roo.dd.Registry
21345  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21346  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21347  * @singleton
21348  */
21349 Roo.dd.Registry = function(){
21350     var elements = {}; 
21351     var handles = {}; 
21352     var autoIdSeed = 0;
21353
21354     var getId = function(el, autogen){
21355         if(typeof el == "string"){
21356             return el;
21357         }
21358         var id = el.id;
21359         if(!id && autogen !== false){
21360             id = "roodd-" + (++autoIdSeed);
21361             el.id = id;
21362         }
21363         return id;
21364     };
21365     
21366     return {
21367     /**
21368      * Register a drag drop element
21369      * @param {String|HTMLElement} element The id or DOM node to register
21370      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21371      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21372      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21373      * populated in the data object (if applicable):
21374      * <pre>
21375 Value      Description<br />
21376 ---------  ------------------------------------------<br />
21377 handles    Array of DOM nodes that trigger dragging<br />
21378            for the element being registered<br />
21379 isHandle   True if the element passed in triggers<br />
21380            dragging itself, else false
21381 </pre>
21382      */
21383         register : function(el, data){
21384             data = data || {};
21385             if(typeof el == "string"){
21386                 el = document.getElementById(el);
21387             }
21388             data.ddel = el;
21389             elements[getId(el)] = data;
21390             if(data.isHandle !== false){
21391                 handles[data.ddel.id] = data;
21392             }
21393             if(data.handles){
21394                 var hs = data.handles;
21395                 for(var i = 0, len = hs.length; i < len; i++){
21396                         handles[getId(hs[i])] = data;
21397                 }
21398             }
21399         },
21400
21401     /**
21402      * Unregister a drag drop element
21403      * @param {String|HTMLElement}  element The id or DOM node to unregister
21404      */
21405         unregister : function(el){
21406             var id = getId(el, false);
21407             var data = elements[id];
21408             if(data){
21409                 delete elements[id];
21410                 if(data.handles){
21411                     var hs = data.handles;
21412                     for(var i = 0, len = hs.length; i < len; i++){
21413                         delete handles[getId(hs[i], false)];
21414                     }
21415                 }
21416             }
21417         },
21418
21419     /**
21420      * Returns the handle registered for a DOM Node by id
21421      * @param {String|HTMLElement} id The DOM node or id to look up
21422      * @return {Object} handle The custom handle data
21423      */
21424         getHandle : function(id){
21425             if(typeof id != "string"){ // must be element?
21426                 id = id.id;
21427             }
21428             return handles[id];
21429         },
21430
21431     /**
21432      * Returns the handle that is registered for the DOM node that is the target of the event
21433      * @param {Event} e The event
21434      * @return {Object} handle The custom handle data
21435      */
21436         getHandleFromEvent : function(e){
21437             var t = Roo.lib.Event.getTarget(e);
21438             return t ? handles[t.id] : null;
21439         },
21440
21441     /**
21442      * Returns a custom data object that is registered for a DOM node by id
21443      * @param {String|HTMLElement} id The DOM node or id to look up
21444      * @return {Object} data The custom data
21445      */
21446         getTarget : function(id){
21447             if(typeof id != "string"){ // must be element?
21448                 id = id.id;
21449             }
21450             return elements[id];
21451         },
21452
21453     /**
21454      * Returns a custom data object that is registered for the DOM node that is the target of the event
21455      * @param {Event} e The event
21456      * @return {Object} data The custom data
21457      */
21458         getTargetFromEvent : function(e){
21459             var t = Roo.lib.Event.getTarget(e);
21460             return t ? elements[t.id] || handles[t.id] : null;
21461         }
21462     };
21463 }();/*
21464  * Based on:
21465  * Ext JS Library 1.1.1
21466  * Copyright(c) 2006-2007, Ext JS, LLC.
21467  *
21468  * Originally Released Under LGPL - original licence link has changed is not relivant.
21469  *
21470  * Fork - LGPL
21471  * <script type="text/javascript">
21472  */
21473  
21474
21475 /**
21476  * @class Roo.dd.StatusProxy
21477  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21478  * default drag proxy used by all Roo.dd components.
21479  * @constructor
21480  * @param {Object} config
21481  */
21482 Roo.dd.StatusProxy = function(config){
21483     Roo.apply(this, config);
21484     this.id = this.id || Roo.id();
21485     this.el = new Roo.Layer({
21486         dh: {
21487             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21488                 {tag: "div", cls: "x-dd-drop-icon"},
21489                 {tag: "div", cls: "x-dd-drag-ghost"}
21490             ]
21491         }, 
21492         shadow: !config || config.shadow !== false
21493     });
21494     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21495     this.dropStatus = this.dropNotAllowed;
21496 };
21497
21498 Roo.dd.StatusProxy.prototype = {
21499     /**
21500      * @cfg {String} dropAllowed
21501      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21502      */
21503     dropAllowed : "x-dd-drop-ok",
21504     /**
21505      * @cfg {String} dropNotAllowed
21506      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21507      */
21508     dropNotAllowed : "x-dd-drop-nodrop",
21509
21510     /**
21511      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21512      * over the current target element.
21513      * @param {String} cssClass The css class for the new drop status indicator image
21514      */
21515     setStatus : function(cssClass){
21516         cssClass = cssClass || this.dropNotAllowed;
21517         if(this.dropStatus != cssClass){
21518             this.el.replaceClass(this.dropStatus, cssClass);
21519             this.dropStatus = cssClass;
21520         }
21521     },
21522
21523     /**
21524      * Resets the status indicator to the default dropNotAllowed value
21525      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21526      */
21527     reset : function(clearGhost){
21528         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21529         this.dropStatus = this.dropNotAllowed;
21530         if(clearGhost){
21531             this.ghost.update("");
21532         }
21533     },
21534
21535     /**
21536      * Updates the contents of the ghost element
21537      * @param {String} html The html that will replace the current innerHTML of the ghost element
21538      */
21539     update : function(html){
21540         if(typeof html == "string"){
21541             this.ghost.update(html);
21542         }else{
21543             this.ghost.update("");
21544             html.style.margin = "0";
21545             this.ghost.dom.appendChild(html);
21546         }
21547         // ensure float = none set?? cant remember why though.
21548         var el = this.ghost.dom.firstChild;
21549                 if(el){
21550                         Roo.fly(el).setStyle('float', 'none');
21551                 }
21552     },
21553     
21554     /**
21555      * Returns the underlying proxy {@link Roo.Layer}
21556      * @return {Roo.Layer} el
21557     */
21558     getEl : function(){
21559         return this.el;
21560     },
21561
21562     /**
21563      * Returns the ghost element
21564      * @return {Roo.Element} el
21565      */
21566     getGhost : function(){
21567         return this.ghost;
21568     },
21569
21570     /**
21571      * Hides the proxy
21572      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21573      */
21574     hide : function(clear){
21575         this.el.hide();
21576         if(clear){
21577             this.reset(true);
21578         }
21579     },
21580
21581     /**
21582      * Stops the repair animation if it's currently running
21583      */
21584     stop : function(){
21585         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21586             this.anim.stop();
21587         }
21588     },
21589
21590     /**
21591      * Displays this proxy
21592      */
21593     show : function(){
21594         this.el.show();
21595     },
21596
21597     /**
21598      * Force the Layer to sync its shadow and shim positions to the element
21599      */
21600     sync : function(){
21601         this.el.sync();
21602     },
21603
21604     /**
21605      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21606      * invalid drop operation by the item being dragged.
21607      * @param {Array} xy The XY position of the element ([x, y])
21608      * @param {Function} callback The function to call after the repair is complete
21609      * @param {Object} scope The scope in which to execute the callback
21610      */
21611     repair : function(xy, callback, scope){
21612         this.callback = callback;
21613         this.scope = scope;
21614         if(xy && this.animRepair !== false){
21615             this.el.addClass("x-dd-drag-repair");
21616             this.el.hideUnders(true);
21617             this.anim = this.el.shift({
21618                 duration: this.repairDuration || .5,
21619                 easing: 'easeOut',
21620                 xy: xy,
21621                 stopFx: true,
21622                 callback: this.afterRepair,
21623                 scope: this
21624             });
21625         }else{
21626             this.afterRepair();
21627         }
21628     },
21629
21630     // private
21631     afterRepair : function(){
21632         this.hide(true);
21633         if(typeof this.callback == "function"){
21634             this.callback.call(this.scope || this);
21635         }
21636         this.callback = null;
21637         this.scope = null;
21638     }
21639 };/*
21640  * Based on:
21641  * Ext JS Library 1.1.1
21642  * Copyright(c) 2006-2007, Ext JS, LLC.
21643  *
21644  * Originally Released Under LGPL - original licence link has changed is not relivant.
21645  *
21646  * Fork - LGPL
21647  * <script type="text/javascript">
21648  */
21649
21650 /**
21651  * @class Roo.dd.DragSource
21652  * @extends Roo.dd.DDProxy
21653  * A simple class that provides the basic implementation needed to make any element draggable.
21654  * @constructor
21655  * @param {String/HTMLElement/Element} el The container element
21656  * @param {Object} config
21657  */
21658 Roo.dd.DragSource = function(el, config){
21659     this.el = Roo.get(el);
21660     this.dragData = {};
21661     
21662     Roo.apply(this, config);
21663     
21664     if(!this.proxy){
21665         this.proxy = new Roo.dd.StatusProxy();
21666     }
21667
21668     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21669           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21670     
21671     this.dragging = false;
21672 };
21673
21674 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21675     /**
21676      * @cfg {String} dropAllowed
21677      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21678      */
21679     dropAllowed : "x-dd-drop-ok",
21680     /**
21681      * @cfg {String} dropNotAllowed
21682      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21683      */
21684     dropNotAllowed : "x-dd-drop-nodrop",
21685
21686     /**
21687      * Returns the data object associated with this drag source
21688      * @return {Object} data An object containing arbitrary data
21689      */
21690     getDragData : function(e){
21691         return this.dragData;
21692     },
21693
21694     // private
21695     onDragEnter : function(e, id){
21696         var target = Roo.dd.DragDropMgr.getDDById(id);
21697         this.cachedTarget = target;
21698         if(this.beforeDragEnter(target, e, id) !== false){
21699             if(target.isNotifyTarget){
21700                 var status = target.notifyEnter(this, e, this.dragData);
21701                 this.proxy.setStatus(status);
21702             }else{
21703                 this.proxy.setStatus(this.dropAllowed);
21704             }
21705             
21706             if(this.afterDragEnter){
21707                 /**
21708                  * An empty function by default, but provided so that you can perform a custom action
21709                  * when the dragged item enters the drop target by providing an implementation.
21710                  * @param {Roo.dd.DragDrop} target The drop target
21711                  * @param {Event} e The event object
21712                  * @param {String} id The id of the dragged element
21713                  * @method afterDragEnter
21714                  */
21715                 this.afterDragEnter(target, e, id);
21716             }
21717         }
21718     },
21719
21720     /**
21721      * An empty function by default, but provided so that you can perform a custom action
21722      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21723      * @param {Roo.dd.DragDrop} target The drop target
21724      * @param {Event} e The event object
21725      * @param {String} id The id of the dragged element
21726      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21727      */
21728     beforeDragEnter : function(target, e, id){
21729         return true;
21730     },
21731
21732     // private
21733     alignElWithMouse: function() {
21734         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21735         this.proxy.sync();
21736     },
21737
21738     // private
21739     onDragOver : function(e, id){
21740         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21741         if(this.beforeDragOver(target, e, id) !== false){
21742             if(target.isNotifyTarget){
21743                 var status = target.notifyOver(this, e, this.dragData);
21744                 this.proxy.setStatus(status);
21745             }
21746
21747             if(this.afterDragOver){
21748                 /**
21749                  * An empty function by default, but provided so that you can perform a custom action
21750                  * while the dragged item is over the drop target by providing an implementation.
21751                  * @param {Roo.dd.DragDrop} target The drop target
21752                  * @param {Event} e The event object
21753                  * @param {String} id The id of the dragged element
21754                  * @method afterDragOver
21755                  */
21756                 this.afterDragOver(target, e, id);
21757             }
21758         }
21759     },
21760
21761     /**
21762      * An empty function by default, but provided so that you can perform a custom action
21763      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21764      * @param {Roo.dd.DragDrop} target The drop target
21765      * @param {Event} e The event object
21766      * @param {String} id The id of the dragged element
21767      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21768      */
21769     beforeDragOver : function(target, e, id){
21770         return true;
21771     },
21772
21773     // private
21774     onDragOut : function(e, id){
21775         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21776         if(this.beforeDragOut(target, e, id) !== false){
21777             if(target.isNotifyTarget){
21778                 target.notifyOut(this, e, this.dragData);
21779             }
21780             this.proxy.reset();
21781             if(this.afterDragOut){
21782                 /**
21783                  * An empty function by default, but provided so that you can perform a custom action
21784                  * after the dragged item is dragged out of the target without dropping.
21785                  * @param {Roo.dd.DragDrop} target The drop target
21786                  * @param {Event} e The event object
21787                  * @param {String} id The id of the dragged element
21788                  * @method afterDragOut
21789                  */
21790                 this.afterDragOut(target, e, id);
21791             }
21792         }
21793         this.cachedTarget = null;
21794     },
21795
21796     /**
21797      * An empty function by default, but provided so that you can perform a custom action before the dragged
21798      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21799      * @param {Roo.dd.DragDrop} target The drop target
21800      * @param {Event} e The event object
21801      * @param {String} id The id of the dragged element
21802      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21803      */
21804     beforeDragOut : function(target, e, id){
21805         return true;
21806     },
21807     
21808     // private
21809     onDragDrop : function(e, id){
21810         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21811         if(this.beforeDragDrop(target, e, id) !== false){
21812             if(target.isNotifyTarget){
21813                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21814                     this.onValidDrop(target, e, id);
21815                 }else{
21816                     this.onInvalidDrop(target, e, id);
21817                 }
21818             }else{
21819                 this.onValidDrop(target, e, id);
21820             }
21821             
21822             if(this.afterDragDrop){
21823                 /**
21824                  * An empty function by default, but provided so that you can perform a custom action
21825                  * after a valid drag drop has occurred by providing an implementation.
21826                  * @param {Roo.dd.DragDrop} target The drop target
21827                  * @param {Event} e The event object
21828                  * @param {String} id The id of the dropped element
21829                  * @method afterDragDrop
21830                  */
21831                 this.afterDragDrop(target, e, id);
21832             }
21833         }
21834         delete this.cachedTarget;
21835     },
21836
21837     /**
21838      * An empty function by default, but provided so that you can perform a custom action before the dragged
21839      * item is dropped onto the target and optionally cancel the onDragDrop.
21840      * @param {Roo.dd.DragDrop} target The drop target
21841      * @param {Event} e The event object
21842      * @param {String} id The id of the dragged element
21843      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21844      */
21845     beforeDragDrop : function(target, e, id){
21846         return true;
21847     },
21848
21849     // private
21850     onValidDrop : function(target, e, id){
21851         this.hideProxy();
21852         if(this.afterValidDrop){
21853             /**
21854              * An empty function by default, but provided so that you can perform a custom action
21855              * after a valid drop has occurred by providing an implementation.
21856              * @param {Object} target The target DD 
21857              * @param {Event} e The event object
21858              * @param {String} id The id of the dropped element
21859              * @method afterInvalidDrop
21860              */
21861             this.afterValidDrop(target, e, id);
21862         }
21863     },
21864
21865     // private
21866     getRepairXY : function(e, data){
21867         return this.el.getXY();  
21868     },
21869
21870     // private
21871     onInvalidDrop : function(target, e, id){
21872         this.beforeInvalidDrop(target, e, id);
21873         if(this.cachedTarget){
21874             if(this.cachedTarget.isNotifyTarget){
21875                 this.cachedTarget.notifyOut(this, e, this.dragData);
21876             }
21877             this.cacheTarget = null;
21878         }
21879         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21880
21881         if(this.afterInvalidDrop){
21882             /**
21883              * An empty function by default, but provided so that you can perform a custom action
21884              * after an invalid drop has occurred by providing an implementation.
21885              * @param {Event} e The event object
21886              * @param {String} id The id of the dropped element
21887              * @method afterInvalidDrop
21888              */
21889             this.afterInvalidDrop(e, id);
21890         }
21891     },
21892
21893     // private
21894     afterRepair : function(){
21895         if(Roo.enableFx){
21896             this.el.highlight(this.hlColor || "c3daf9");
21897         }
21898         this.dragging = false;
21899     },
21900
21901     /**
21902      * An empty function by default, but provided so that you can perform a custom action after an invalid
21903      * drop has occurred.
21904      * @param {Roo.dd.DragDrop} target The drop target
21905      * @param {Event} e The event object
21906      * @param {String} id The id of the dragged element
21907      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21908      */
21909     beforeInvalidDrop : function(target, e, id){
21910         return true;
21911     },
21912
21913     // private
21914     handleMouseDown : function(e){
21915         if(this.dragging) {
21916             return;
21917         }
21918         var data = this.getDragData(e);
21919         if(data && this.onBeforeDrag(data, e) !== false){
21920             this.dragData = data;
21921             this.proxy.stop();
21922             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21923         } 
21924     },
21925
21926     /**
21927      * An empty function by default, but provided so that you can perform a custom action before the initial
21928      * drag event begins and optionally cancel it.
21929      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21930      * @param {Event} e The event object
21931      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21932      */
21933     onBeforeDrag : function(data, e){
21934         return true;
21935     },
21936
21937     /**
21938      * An empty function by default, but provided so that you can perform a custom action once the initial
21939      * drag event has begun.  The drag cannot be canceled from this function.
21940      * @param {Number} x The x position of the click on the dragged object
21941      * @param {Number} y The y position of the click on the dragged object
21942      */
21943     onStartDrag : Roo.emptyFn,
21944
21945     // private - YUI override
21946     startDrag : function(x, y){
21947         this.proxy.reset();
21948         this.dragging = true;
21949         this.proxy.update("");
21950         this.onInitDrag(x, y);
21951         this.proxy.show();
21952     },
21953
21954     // private
21955     onInitDrag : function(x, y){
21956         var clone = this.el.dom.cloneNode(true);
21957         clone.id = Roo.id(); // prevent duplicate ids
21958         this.proxy.update(clone);
21959         this.onStartDrag(x, y);
21960         return true;
21961     },
21962
21963     /**
21964      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21965      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21966      */
21967     getProxy : function(){
21968         return this.proxy;  
21969     },
21970
21971     /**
21972      * Hides the drag source's {@link Roo.dd.StatusProxy}
21973      */
21974     hideProxy : function(){
21975         this.proxy.hide();  
21976         this.proxy.reset(true);
21977         this.dragging = false;
21978     },
21979
21980     // private
21981     triggerCacheRefresh : function(){
21982         Roo.dd.DDM.refreshCache(this.groups);
21983     },
21984
21985     // private - override to prevent hiding
21986     b4EndDrag: function(e) {
21987     },
21988
21989     // private - override to prevent moving
21990     endDrag : function(e){
21991         this.onEndDrag(this.dragData, e);
21992     },
21993
21994     // private
21995     onEndDrag : function(data, e){
21996     },
21997     
21998     // private - pin to cursor
21999     autoOffset : function(x, y) {
22000         this.setDelta(-12, -20);
22001     }    
22002 });/*
22003  * Based on:
22004  * Ext JS Library 1.1.1
22005  * Copyright(c) 2006-2007, Ext JS, LLC.
22006  *
22007  * Originally Released Under LGPL - original licence link has changed is not relivant.
22008  *
22009  * Fork - LGPL
22010  * <script type="text/javascript">
22011  */
22012
22013
22014 /**
22015  * @class Roo.dd.DropTarget
22016  * @extends Roo.dd.DDTarget
22017  * A simple class that provides the basic implementation needed to make any element a drop target that can have
22018  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
22019  * @constructor
22020  * @param {String/HTMLElement/Element} el The container element
22021  * @param {Object} config
22022  */
22023 Roo.dd.DropTarget = function(el, config){
22024     this.el = Roo.get(el);
22025     
22026     var listeners = false; ;
22027     if (config && config.listeners) {
22028         listeners= config.listeners;
22029         delete config.listeners;
22030     }
22031     Roo.apply(this, config);
22032     
22033     if(this.containerScroll){
22034         Roo.dd.ScrollManager.register(this.el);
22035     }
22036     this.addEvents( {
22037          /**
22038          * @scope Roo.dd.DropTarget
22039          */
22040          
22041          /**
22042          * @event enter
22043          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22044          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22045          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22046          * 
22047          * IMPORTANT : it should set this.overClass and this.dropAllowed
22048          * 
22049          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22050          * @param {Event} e The event
22051          * @param {Object} data An object containing arbitrary data supplied by the drag source
22052          */
22053         "enter" : true,
22054         
22055          /**
22056          * @event over
22057          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22058          * This method will be called on every mouse movement while the drag source is over the drop target.
22059          * This default implementation simply returns the dropAllowed config value.
22060          * 
22061          * IMPORTANT : it should set this.dropAllowed
22062          * 
22063          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22064          * @param {Event} e The event
22065          * @param {Object} data An object containing arbitrary data supplied by the drag source
22066          
22067          */
22068         "over" : true,
22069         /**
22070          * @event out
22071          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22072          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22073          * overClass (if any) from the drop element.
22074          * 
22075          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22076          * @param {Event} e The event
22077          * @param {Object} data An object containing arbitrary data supplied by the drag source
22078          */
22079          "out" : true,
22080          
22081         /**
22082          * @event drop
22083          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22084          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22085          * implementation that does something to process the drop event and returns true so that the drag source's
22086          * repair action does not run.
22087          * 
22088          * IMPORTANT : it should set this.success
22089          * 
22090          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22091          * @param {Event} e The event
22092          * @param {Object} data An object containing arbitrary data supplied by the drag source
22093         */
22094          "drop" : true
22095     });
22096             
22097      
22098     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22099         this.el.dom, 
22100         this.ddGroup || this.group,
22101         {
22102             isTarget: true,
22103             listeners : listeners || {} 
22104            
22105         
22106         }
22107     );
22108
22109 };
22110
22111 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22112     /**
22113      * @cfg {String} overClass
22114      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22115      */
22116      /**
22117      * @cfg {String} ddGroup
22118      * The drag drop group to handle drop events for
22119      */
22120      
22121     /**
22122      * @cfg {String} dropAllowed
22123      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22124      */
22125     dropAllowed : "x-dd-drop-ok",
22126     /**
22127      * @cfg {String} dropNotAllowed
22128      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22129      */
22130     dropNotAllowed : "x-dd-drop-nodrop",
22131     /**
22132      * @cfg {boolean} success
22133      * set this after drop listener.. 
22134      */
22135     success : false,
22136     /**
22137      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22138      * if the drop point is valid for over/enter..
22139      */
22140     valid : false,
22141     // private
22142     isTarget : true,
22143
22144     // private
22145     isNotifyTarget : true,
22146     
22147     /**
22148      * @hide
22149      */
22150     notifyEnter : function(dd, e, data)
22151     {
22152         this.valid = true;
22153         this.fireEvent('enter', dd, e, data);
22154         if(this.overClass){
22155             this.el.addClass(this.overClass);
22156         }
22157         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22158             this.valid ? this.dropAllowed : this.dropNotAllowed
22159         );
22160     },
22161
22162     /**
22163      * @hide
22164      */
22165     notifyOver : function(dd, e, data)
22166     {
22167         this.valid = true;
22168         this.fireEvent('over', dd, e, data);
22169         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22170             this.valid ? this.dropAllowed : this.dropNotAllowed
22171         );
22172     },
22173
22174     /**
22175      * @hide
22176      */
22177     notifyOut : function(dd, e, data)
22178     {
22179         this.fireEvent('out', dd, e, data);
22180         if(this.overClass){
22181             this.el.removeClass(this.overClass);
22182         }
22183     },
22184
22185     /**
22186      * @hide
22187      */
22188     notifyDrop : function(dd, e, data)
22189     {
22190         this.success = false;
22191         this.fireEvent('drop', dd, e, data);
22192         return this.success;
22193     }
22194 });/*
22195  * Based on:
22196  * Ext JS Library 1.1.1
22197  * Copyright(c) 2006-2007, Ext JS, LLC.
22198  *
22199  * Originally Released Under LGPL - original licence link has changed is not relivant.
22200  *
22201  * Fork - LGPL
22202  * <script type="text/javascript">
22203  */
22204
22205
22206 /**
22207  * @class Roo.dd.DragZone
22208  * @extends Roo.dd.DragSource
22209  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22210  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22211  * @constructor
22212  * @param {String/HTMLElement/Element} el The container element
22213  * @param {Object} config
22214  */
22215 Roo.dd.DragZone = function(el, config){
22216     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22217     if(this.containerScroll){
22218         Roo.dd.ScrollManager.register(this.el);
22219     }
22220 };
22221
22222 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22223     /**
22224      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22225      * for auto scrolling during drag operations.
22226      */
22227     /**
22228      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22229      * method after a failed drop (defaults to "c3daf9" - light blue)
22230      */
22231
22232     /**
22233      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22234      * for a valid target to drag based on the mouse down. Override this method
22235      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22236      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22237      * @param {EventObject} e The mouse down event
22238      * @return {Object} The dragData
22239      */
22240     getDragData : function(e){
22241         return Roo.dd.Registry.getHandleFromEvent(e);
22242     },
22243     
22244     /**
22245      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22246      * this.dragData.ddel
22247      * @param {Number} x The x position of the click on the dragged object
22248      * @param {Number} y The y position of the click on the dragged object
22249      * @return {Boolean} true to continue the drag, false to cancel
22250      */
22251     onInitDrag : function(x, y){
22252         this.proxy.update(this.dragData.ddel.cloneNode(true));
22253         this.onStartDrag(x, y);
22254         return true;
22255     },
22256     
22257     /**
22258      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22259      */
22260     afterRepair : function(){
22261         if(Roo.enableFx){
22262             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22263         }
22264         this.dragging = false;
22265     },
22266
22267     /**
22268      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22269      * the XY of this.dragData.ddel
22270      * @param {EventObject} e The mouse up event
22271      * @return {Array} The xy location (e.g. [100, 200])
22272      */
22273     getRepairXY : function(e){
22274         return Roo.Element.fly(this.dragData.ddel).getXY();  
22275     }
22276 });/*
22277  * Based on:
22278  * Ext JS Library 1.1.1
22279  * Copyright(c) 2006-2007, Ext JS, LLC.
22280  *
22281  * Originally Released Under LGPL - original licence link has changed is not relivant.
22282  *
22283  * Fork - LGPL
22284  * <script type="text/javascript">
22285  */
22286 /**
22287  * @class Roo.dd.DropZone
22288  * @extends Roo.dd.DropTarget
22289  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22290  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22291  * @constructor
22292  * @param {String/HTMLElement/Element} el The container element
22293  * @param {Object} config
22294  */
22295 Roo.dd.DropZone = function(el, config){
22296     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22297 };
22298
22299 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22300     /**
22301      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22302      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22303      * provide your own custom lookup.
22304      * @param {Event} e The event
22305      * @return {Object} data The custom data
22306      */
22307     getTargetFromEvent : function(e){
22308         return Roo.dd.Registry.getTargetFromEvent(e);
22309     },
22310
22311     /**
22312      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22313      * that it has registered.  This method has no default implementation and should be overridden to provide
22314      * node-specific processing if necessary.
22315      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22316      * {@link #getTargetFromEvent} for this node)
22317      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22318      * @param {Event} e The event
22319      * @param {Object} data An object containing arbitrary data supplied by the drag source
22320      */
22321     onNodeEnter : function(n, dd, e, data){
22322         
22323     },
22324
22325     /**
22326      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22327      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22328      * overridden to provide the proper feedback.
22329      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22330      * {@link #getTargetFromEvent} for this node)
22331      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22332      * @param {Event} e The event
22333      * @param {Object} data An object containing arbitrary data supplied by the drag source
22334      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22335      * underlying {@link Roo.dd.StatusProxy} can be updated
22336      */
22337     onNodeOver : function(n, dd, e, data){
22338         return this.dropAllowed;
22339     },
22340
22341     /**
22342      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22343      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22344      * node-specific processing if necessary.
22345      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22346      * {@link #getTargetFromEvent} for this node)
22347      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22348      * @param {Event} e The event
22349      * @param {Object} data An object containing arbitrary data supplied by the drag source
22350      */
22351     onNodeOut : function(n, dd, e, data){
22352         
22353     },
22354
22355     /**
22356      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22357      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22358      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22359      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22360      * {@link #getTargetFromEvent} for this node)
22361      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22362      * @param {Event} e The event
22363      * @param {Object} data An object containing arbitrary data supplied by the drag source
22364      * @return {Boolean} True if the drop was valid, else false
22365      */
22366     onNodeDrop : function(n, dd, e, data){
22367         return false;
22368     },
22369
22370     /**
22371      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22372      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22373      * it should be overridden to provide the proper feedback if necessary.
22374      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22375      * @param {Event} e The event
22376      * @param {Object} data An object containing arbitrary data supplied by the drag source
22377      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22378      * underlying {@link Roo.dd.StatusProxy} can be updated
22379      */
22380     onContainerOver : function(dd, e, data){
22381         return this.dropNotAllowed;
22382     },
22383
22384     /**
22385      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22386      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22387      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22388      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22389      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22390      * @param {Event} e The event
22391      * @param {Object} data An object containing arbitrary data supplied by the drag source
22392      * @return {Boolean} True if the drop was valid, else false
22393      */
22394     onContainerDrop : function(dd, e, data){
22395         return false;
22396     },
22397
22398     /**
22399      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22400      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22401      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22402      * you should override this method and provide a custom implementation.
22403      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22404      * @param {Event} e The event
22405      * @param {Object} data An object containing arbitrary data supplied by the drag source
22406      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22407      * underlying {@link Roo.dd.StatusProxy} can be updated
22408      */
22409     notifyEnter : function(dd, e, data){
22410         return this.dropNotAllowed;
22411     },
22412
22413     /**
22414      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22415      * This method will be called on every mouse movement while the drag source is over the drop zone.
22416      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22417      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22418      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22419      * registered node, it will call {@link #onContainerOver}.
22420      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22421      * @param {Event} e The event
22422      * @param {Object} data An object containing arbitrary data supplied by the drag source
22423      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22424      * underlying {@link Roo.dd.StatusProxy} can be updated
22425      */
22426     notifyOver : function(dd, e, data){
22427         var n = this.getTargetFromEvent(e);
22428         if(!n){ // not over valid drop target
22429             if(this.lastOverNode){
22430                 this.onNodeOut(this.lastOverNode, dd, e, data);
22431                 this.lastOverNode = null;
22432             }
22433             return this.onContainerOver(dd, e, data);
22434         }
22435         if(this.lastOverNode != n){
22436             if(this.lastOverNode){
22437                 this.onNodeOut(this.lastOverNode, dd, e, data);
22438             }
22439             this.onNodeEnter(n, dd, e, data);
22440             this.lastOverNode = n;
22441         }
22442         return this.onNodeOver(n, dd, e, data);
22443     },
22444
22445     /**
22446      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22447      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22448      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22449      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22450      * @param {Event} e The event
22451      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22452      */
22453     notifyOut : function(dd, e, data){
22454         if(this.lastOverNode){
22455             this.onNodeOut(this.lastOverNode, dd, e, data);
22456             this.lastOverNode = null;
22457         }
22458     },
22459
22460     /**
22461      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22462      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22463      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22464      * otherwise it will call {@link #onContainerDrop}.
22465      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22466      * @param {Event} e The event
22467      * @param {Object} data An object containing arbitrary data supplied by the drag source
22468      * @return {Boolean} True if the drop was valid, else false
22469      */
22470     notifyDrop : function(dd, e, data){
22471         if(this.lastOverNode){
22472             this.onNodeOut(this.lastOverNode, dd, e, data);
22473             this.lastOverNode = null;
22474         }
22475         var n = this.getTargetFromEvent(e);
22476         return n ?
22477             this.onNodeDrop(n, dd, e, data) :
22478             this.onContainerDrop(dd, e, data);
22479     },
22480
22481     // private
22482     triggerCacheRefresh : function(){
22483         Roo.dd.DDM.refreshCache(this.groups);
22484     }  
22485 });/*
22486  * Based on:
22487  * Ext JS Library 1.1.1
22488  * Copyright(c) 2006-2007, Ext JS, LLC.
22489  *
22490  * Originally Released Under LGPL - original licence link has changed is not relivant.
22491  *
22492  * Fork - LGPL
22493  * <script type="text/javascript">
22494  */
22495
22496
22497 /**
22498  * @class Roo.data.SortTypes
22499  * @singleton
22500  * Defines the default sorting (casting?) comparison functions used when sorting data.
22501  */
22502 Roo.data.SortTypes = {
22503     /**
22504      * Default sort that does nothing
22505      * @param {Mixed} s The value being converted
22506      * @return {Mixed} The comparison value
22507      */
22508     none : function(s){
22509         return s;
22510     },
22511     
22512     /**
22513      * The regular expression used to strip tags
22514      * @type {RegExp}
22515      * @property
22516      */
22517     stripTagsRE : /<\/?[^>]+>/gi,
22518     
22519     /**
22520      * Strips all HTML tags to sort on text only
22521      * @param {Mixed} s The value being converted
22522      * @return {String} The comparison value
22523      */
22524     asText : function(s){
22525         return String(s).replace(this.stripTagsRE, "");
22526     },
22527     
22528     /**
22529      * Strips all HTML tags to sort on text only - Case insensitive
22530      * @param {Mixed} s The value being converted
22531      * @return {String} The comparison value
22532      */
22533     asUCText : function(s){
22534         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22535     },
22536     
22537     /**
22538      * Case insensitive string
22539      * @param {Mixed} s The value being converted
22540      * @return {String} The comparison value
22541      */
22542     asUCString : function(s) {
22543         return String(s).toUpperCase();
22544     },
22545     
22546     /**
22547      * Date sorting
22548      * @param {Mixed} s The value being converted
22549      * @return {Number} The comparison value
22550      */
22551     asDate : function(s) {
22552         if(!s){
22553             return 0;
22554         }
22555         if(s instanceof Date){
22556             return s.getTime();
22557         }
22558         return Date.parse(String(s));
22559     },
22560     
22561     /**
22562      * Float sorting
22563      * @param {Mixed} s The value being converted
22564      * @return {Float} The comparison value
22565      */
22566     asFloat : function(s) {
22567         var val = parseFloat(String(s).replace(/,/g, ""));
22568         if(isNaN(val)) {
22569             val = 0;
22570         }
22571         return val;
22572     },
22573     
22574     /**
22575      * Integer sorting
22576      * @param {Mixed} s The value being converted
22577      * @return {Number} The comparison value
22578      */
22579     asInt : function(s) {
22580         var val = parseInt(String(s).replace(/,/g, ""));
22581         if(isNaN(val)) {
22582             val = 0;
22583         }
22584         return val;
22585     }
22586 };/*
22587  * Based on:
22588  * Ext JS Library 1.1.1
22589  * Copyright(c) 2006-2007, Ext JS, LLC.
22590  *
22591  * Originally Released Under LGPL - original licence link has changed is not relivant.
22592  *
22593  * Fork - LGPL
22594  * <script type="text/javascript">
22595  */
22596
22597 /**
22598 * @class Roo.data.Record
22599  * Instances of this class encapsulate both record <em>definition</em> information, and record
22600  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22601  * to access Records cached in an {@link Roo.data.Store} object.<br>
22602  * <p>
22603  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22604  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22605  * objects.<br>
22606  * <p>
22607  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22608  * @constructor
22609  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22610  * {@link #create}. The parameters are the same.
22611  * @param {Array} data An associative Array of data values keyed by the field name.
22612  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22613  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22614  * not specified an integer id is generated.
22615  */
22616 Roo.data.Record = function(data, id){
22617     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22618     this.data = data;
22619 };
22620
22621 /**
22622  * Generate a constructor for a specific record layout.
22623  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22624  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22625  * Each field definition object may contain the following properties: <ul>
22626  * <li><b>name</b> : String<p style="margin-left:1em">The name by which the field is referenced within the Record. This is referenced by,
22627  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22628  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22629  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22630  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22631  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22632  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22633  * this may be omitted.</p></li>
22634  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22635  * <ul><li>auto (Default, implies no conversion)</li>
22636  * <li>string</li>
22637  * <li>int</li>
22638  * <li>float</li>
22639  * <li>boolean</li>
22640  * <li>date</li></ul></p></li>
22641  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22642  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22643  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22644  * by the Reader into an object that will be stored in the Record. It is passed the
22645  * following parameters:<ul>
22646  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22647  * </ul></p></li>
22648  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22649  * </ul>
22650  * <br>usage:<br><pre><code>
22651 var TopicRecord = Roo.data.Record.create(
22652     {name: 'title', mapping: 'topic_title'},
22653     {name: 'author', mapping: 'username'},
22654     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22655     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22656     {name: 'lastPoster', mapping: 'user2'},
22657     {name: 'excerpt', mapping: 'post_text'}
22658 );
22659
22660 var myNewRecord = new TopicRecord({
22661     title: 'Do my job please',
22662     author: 'noobie',
22663     totalPosts: 1,
22664     lastPost: new Date(),
22665     lastPoster: 'Animal',
22666     excerpt: 'No way dude!'
22667 });
22668 myStore.add(myNewRecord);
22669 </code></pre>
22670  * @method create
22671  * @static
22672  */
22673 Roo.data.Record.create = function(o){
22674     var f = function(){
22675         f.superclass.constructor.apply(this, arguments);
22676     };
22677     Roo.extend(f, Roo.data.Record);
22678     var p = f.prototype;
22679     p.fields = new Roo.util.MixedCollection(false, function(field){
22680         return field.name;
22681     });
22682     for(var i = 0, len = o.length; i < len; i++){
22683         p.fields.add(new Roo.data.Field(o[i]));
22684     }
22685     f.getField = function(name){
22686         return p.fields.get(name);  
22687     };
22688     return f;
22689 };
22690
22691 Roo.data.Record.AUTO_ID = 1000;
22692 Roo.data.Record.EDIT = 'edit';
22693 Roo.data.Record.REJECT = 'reject';
22694 Roo.data.Record.COMMIT = 'commit';
22695
22696 Roo.data.Record.prototype = {
22697     /**
22698      * Readonly flag - true if this record has been modified.
22699      * @type Boolean
22700      */
22701     dirty : false,
22702     editing : false,
22703     error: null,
22704     modified: null,
22705
22706     // private
22707     join : function(store){
22708         this.store = store;
22709     },
22710
22711     /**
22712      * Set the named field to the specified value.
22713      * @param {String} name The name of the field to set.
22714      * @param {Object} value The value to set the field to.
22715      */
22716     set : function(name, value){
22717         if(this.data[name] == value){
22718             return;
22719         }
22720         this.dirty = true;
22721         if(!this.modified){
22722             this.modified = {};
22723         }
22724         if(typeof this.modified[name] == 'undefined'){
22725             this.modified[name] = this.data[name];
22726         }
22727         this.data[name] = value;
22728         if(!this.editing && this.store){
22729             this.store.afterEdit(this);
22730         }       
22731     },
22732
22733     /**
22734      * Get the value of the named field.
22735      * @param {String} name The name of the field to get the value of.
22736      * @return {Object} The value of the field.
22737      */
22738     get : function(name){
22739         return this.data[name]; 
22740     },
22741
22742     // private
22743     beginEdit : function(){
22744         this.editing = true;
22745         this.modified = {}; 
22746     },
22747
22748     // private
22749     cancelEdit : function(){
22750         this.editing = false;
22751         delete this.modified;
22752     },
22753
22754     // private
22755     endEdit : function(){
22756         this.editing = false;
22757         if(this.dirty && this.store){
22758             this.store.afterEdit(this);
22759         }
22760     },
22761
22762     /**
22763      * Usually called by the {@link Roo.data.Store} which owns the Record.
22764      * Rejects all changes made to the Record since either creation, or the last commit operation.
22765      * Modified fields are reverted to their original values.
22766      * <p>
22767      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22768      * of reject operations.
22769      */
22770     reject : function(){
22771         var m = this.modified;
22772         for(var n in m){
22773             if(typeof m[n] != "function"){
22774                 this.data[n] = m[n];
22775             }
22776         }
22777         this.dirty = false;
22778         delete this.modified;
22779         this.editing = false;
22780         if(this.store){
22781             this.store.afterReject(this);
22782         }
22783     },
22784
22785     /**
22786      * Usually called by the {@link Roo.data.Store} which owns the Record.
22787      * Commits all changes made to the Record since either creation, or the last commit operation.
22788      * <p>
22789      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22790      * of commit operations.
22791      */
22792     commit : function(){
22793         this.dirty = false;
22794         delete this.modified;
22795         this.editing = false;
22796         if(this.store){
22797             this.store.afterCommit(this);
22798         }
22799     },
22800
22801     // private
22802     hasError : function(){
22803         return this.error != null;
22804     },
22805
22806     // private
22807     clearError : function(){
22808         this.error = null;
22809     },
22810
22811     /**
22812      * Creates a copy of this record.
22813      * @param {String} id (optional) A new record id if you don't want to use this record's id
22814      * @return {Record}
22815      */
22816     copy : function(newId) {
22817         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22818     }
22819 };/*
22820  * Based on:
22821  * Ext JS Library 1.1.1
22822  * Copyright(c) 2006-2007, Ext JS, LLC.
22823  *
22824  * Originally Released Under LGPL - original licence link has changed is not relivant.
22825  *
22826  * Fork - LGPL
22827  * <script type="text/javascript">
22828  */
22829
22830
22831
22832 /**
22833  * @class Roo.data.Store
22834  * @extends Roo.util.Observable
22835  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22836  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22837  * <p>
22838  * A Store object uses an implementation of {@link Roo.data.DataProxy} to access a data object unless you call loadData() directly and pass in your data. The Store object
22839  * has no knowledge of the format of the data returned by the Proxy.<br>
22840  * <p>
22841  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22842  * instances from the data object. These records are cached and made available through accessor functions.
22843  * @constructor
22844  * Creates a new Store.
22845  * @param {Object} config A config object containing the objects needed for the Store to access data,
22846  * and read the data into Records.
22847  */
22848 Roo.data.Store = function(config){
22849     this.data = new Roo.util.MixedCollection(false);
22850     this.data.getKey = function(o){
22851         return o.id;
22852     };
22853     this.baseParams = {};
22854     // private
22855     this.paramNames = {
22856         "start" : "start",
22857         "limit" : "limit",
22858         "sort" : "sort",
22859         "dir" : "dir",
22860         "multisort" : "_multisort"
22861     };
22862
22863     if(config && config.data){
22864         this.inlineData = config.data;
22865         delete config.data;
22866     }
22867
22868     Roo.apply(this, config);
22869     
22870     if(this.reader){ // reader passed
22871         this.reader = Roo.factory(this.reader, Roo.data);
22872         this.reader.xmodule = this.xmodule || false;
22873         if(!this.recordType){
22874             this.recordType = this.reader.recordType;
22875         }
22876         if(this.reader.onMetaChange){
22877             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22878         }
22879     }
22880
22881     if(this.recordType){
22882         this.fields = this.recordType.prototype.fields;
22883     }
22884     this.modified = [];
22885
22886     this.addEvents({
22887         /**
22888          * @event datachanged
22889          * Fires when the data cache has changed, and a widget which is using this Store
22890          * as a Record cache should refresh its view.
22891          * @param {Store} this
22892          */
22893         datachanged : true,
22894         /**
22895          * @event metachange
22896          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22897          * @param {Store} this
22898          * @param {Object} meta The JSON metadata
22899          */
22900         metachange : true,
22901         /**
22902          * @event add
22903          * Fires when Records have been added to the Store
22904          * @param {Store} this
22905          * @param {Roo.data.Record[]} records The array of Records added
22906          * @param {Number} index The index at which the record(s) were added
22907          */
22908         add : true,
22909         /**
22910          * @event remove
22911          * Fires when a Record has been removed from the Store
22912          * @param {Store} this
22913          * @param {Roo.data.Record} record The Record that was removed
22914          * @param {Number} index The index at which the record was removed
22915          */
22916         remove : true,
22917         /**
22918          * @event update
22919          * Fires when a Record has been updated
22920          * @param {Store} this
22921          * @param {Roo.data.Record} record The Record that was updated
22922          * @param {String} operation The update operation being performed.  Value may be one of:
22923          * <pre><code>
22924  Roo.data.Record.EDIT
22925  Roo.data.Record.REJECT
22926  Roo.data.Record.COMMIT
22927          * </code></pre>
22928          */
22929         update : true,
22930         /**
22931          * @event clear
22932          * Fires when the data cache has been cleared.
22933          * @param {Store} this
22934          */
22935         clear : true,
22936         /**
22937          * @event beforeload
22938          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22939          * the load action will be canceled.
22940          * @param {Store} this
22941          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22942          */
22943         beforeload : true,
22944         /**
22945          * @event beforeloadadd
22946          * Fires after a new set of Records has been loaded.
22947          * @param {Store} this
22948          * @param {Roo.data.Record[]} records The Records that were loaded
22949          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22950          */
22951         beforeloadadd : true,
22952         /**
22953          * @event load
22954          * Fires after a new set of Records has been loaded, before they are added to the store.
22955          * @param {Store} this
22956          * @param {Roo.data.Record[]} records The Records that were loaded
22957          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22958          * @params {Object} return from reader
22959          */
22960         load : true,
22961         /**
22962          * @event loadexception
22963          * Fires if an exception occurs in the Proxy during loading.
22964          * Called with the signature of the Proxy's "loadexception" event.
22965          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22966          * 
22967          * @param {Proxy} 
22968          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22969          * @param {Object} load options 
22970          * @param {Object} jsonData from your request (normally this contains the Exception)
22971          */
22972         loadexception : true
22973     });
22974     
22975     if(this.proxy){
22976         this.proxy = Roo.factory(this.proxy, Roo.data);
22977         this.proxy.xmodule = this.xmodule || false;
22978         this.relayEvents(this.proxy,  ["loadexception"]);
22979     }
22980     this.sortToggle = {};
22981     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22982
22983     Roo.data.Store.superclass.constructor.call(this);
22984
22985     if(this.inlineData){
22986         this.loadData(this.inlineData);
22987         delete this.inlineData;
22988     }
22989 };
22990
22991 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22992      /**
22993     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22994     * without a remote query - used by combo/forms at present.
22995     */
22996     
22997     /**
22998     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22999     */
23000     /**
23001     * @cfg {Array} data Inline data to be loaded when the store is initialized.
23002     */
23003     /**
23004     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
23005     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
23006     */
23007     /**
23008     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
23009     * on any HTTP request
23010     */
23011     /**
23012     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
23013     */
23014     /**
23015     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
23016     */
23017     multiSort: false,
23018     /**
23019     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
23020     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
23021     */
23022     remoteSort : false,
23023
23024     /**
23025     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
23026      * loaded or when a record is removed. (defaults to false).
23027     */
23028     pruneModifiedRecords : false,
23029
23030     // private
23031     lastOptions : null,
23032
23033     /**
23034      * Add Records to the Store and fires the add event.
23035      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23036      */
23037     add : function(records){
23038         records = [].concat(records);
23039         for(var i = 0, len = records.length; i < len; i++){
23040             records[i].join(this);
23041         }
23042         var index = this.data.length;
23043         this.data.addAll(records);
23044         this.fireEvent("add", this, records, index);
23045     },
23046
23047     /**
23048      * Remove a Record from the Store and fires the remove event.
23049      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23050      */
23051     remove : function(record){
23052         var index = this.data.indexOf(record);
23053         this.data.removeAt(index);
23054  
23055         if(this.pruneModifiedRecords){
23056             this.modified.remove(record);
23057         }
23058         this.fireEvent("remove", this, record, index);
23059     },
23060
23061     /**
23062      * Remove all Records from the Store and fires the clear event.
23063      */
23064     removeAll : function(){
23065         this.data.clear();
23066         if(this.pruneModifiedRecords){
23067             this.modified = [];
23068         }
23069         this.fireEvent("clear", this);
23070     },
23071
23072     /**
23073      * Inserts Records to the Store at the given index and fires the add event.
23074      * @param {Number} index The start index at which to insert the passed Records.
23075      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23076      */
23077     insert : function(index, records){
23078         records = [].concat(records);
23079         for(var i = 0, len = records.length; i < len; i++){
23080             this.data.insert(index, records[i]);
23081             records[i].join(this);
23082         }
23083         this.fireEvent("add", this, records, index);
23084     },
23085
23086     /**
23087      * Get the index within the cache of the passed Record.
23088      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23089      * @return {Number} The index of the passed Record. Returns -1 if not found.
23090      */
23091     indexOf : function(record){
23092         return this.data.indexOf(record);
23093     },
23094
23095     /**
23096      * Get the index within the cache of the Record with the passed id.
23097      * @param {String} id The id of the Record to find.
23098      * @return {Number} The index of the Record. Returns -1 if not found.
23099      */
23100     indexOfId : function(id){
23101         return this.data.indexOfKey(id);
23102     },
23103
23104     /**
23105      * Get the Record with the specified id.
23106      * @param {String} id The id of the Record to find.
23107      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23108      */
23109     getById : function(id){
23110         return this.data.key(id);
23111     },
23112
23113     /**
23114      * Get the Record at the specified index.
23115      * @param {Number} index The index of the Record to find.
23116      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23117      */
23118     getAt : function(index){
23119         return this.data.itemAt(index);
23120     },
23121
23122     /**
23123      * Returns a range of Records between specified indices.
23124      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23125      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23126      * @return {Roo.data.Record[]} An array of Records
23127      */
23128     getRange : function(start, end){
23129         return this.data.getRange(start, end);
23130     },
23131
23132     // private
23133     storeOptions : function(o){
23134         o = Roo.apply({}, o);
23135         delete o.callback;
23136         delete o.scope;
23137         this.lastOptions = o;
23138     },
23139
23140     /**
23141      * Loads the Record cache from the configured Proxy using the configured Reader.
23142      * <p>
23143      * If using remote paging, then the first load call must specify the <em>start</em>
23144      * and <em>limit</em> properties in the options.params property to establish the initial
23145      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23146      * <p>
23147      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23148      * and this call will return before the new data has been loaded. Perform any post-processing
23149      * in a callback function, or in a "load" event handler.</strong>
23150      * <p>
23151      * @param {Object} options An object containing properties which control loading options:<ul>
23152      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23153      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23154      * passed the following arguments:<ul>
23155      * <li>r : Roo.data.Record[]</li>
23156      * <li>options: Options object from the load call</li>
23157      * <li>success: Boolean success indicator</li></ul></li>
23158      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23159      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23160      * </ul>
23161      */
23162     load : function(options){
23163         options = options || {};
23164         if(this.fireEvent("beforeload", this, options) !== false){
23165             this.storeOptions(options);
23166             var p = Roo.apply(options.params || {}, this.baseParams);
23167             // if meta was not loaded from remote source.. try requesting it.
23168             if (!this.reader.metaFromRemote) {
23169                 p._requestMeta = 1;
23170             }
23171             if(this.sortInfo && this.remoteSort){
23172                 var pn = this.paramNames;
23173                 p[pn["sort"]] = this.sortInfo.field;
23174                 p[pn["dir"]] = this.sortInfo.direction;
23175             }
23176             if (this.multiSort) {
23177                 var pn = this.paramNames;
23178                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23179             }
23180             
23181             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23182         }
23183     },
23184
23185     /**
23186      * Reloads the Record cache from the configured Proxy using the configured Reader and
23187      * the options from the last load operation performed.
23188      * @param {Object} options (optional) An object containing properties which may override the options
23189      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23190      * the most recently used options are reused).
23191      */
23192     reload : function(options){
23193         this.load(Roo.applyIf(options||{}, this.lastOptions));
23194     },
23195
23196     // private
23197     // Called as a callback by the Reader during a load operation.
23198     loadRecords : function(o, options, success){
23199         if(!o || success === false){
23200             if(success !== false){
23201                 this.fireEvent("load", this, [], options, o);
23202             }
23203             if(options.callback){
23204                 options.callback.call(options.scope || this, [], options, false);
23205             }
23206             return;
23207         }
23208         // if data returned failure - throw an exception.
23209         if (o.success === false) {
23210             // show a message if no listener is registered.
23211             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23212                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23213             }
23214             // loadmask wil be hooked into this..
23215             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23216             return;
23217         }
23218         var r = o.records, t = o.totalRecords || r.length;
23219         
23220         this.fireEvent("beforeloadadd", this, r, options, o);
23221         
23222         if(!options || options.add !== true){
23223             if(this.pruneModifiedRecords){
23224                 this.modified = [];
23225             }
23226             for(var i = 0, len = r.length; i < len; i++){
23227                 r[i].join(this);
23228             }
23229             if(this.snapshot){
23230                 this.data = this.snapshot;
23231                 delete this.snapshot;
23232             }
23233             this.data.clear();
23234             this.data.addAll(r);
23235             this.totalLength = t;
23236             this.applySort();
23237             this.fireEvent("datachanged", this);
23238         }else{
23239             this.totalLength = Math.max(t, this.data.length+r.length);
23240             this.add(r);
23241         }
23242         
23243         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23244                 
23245             var e = new Roo.data.Record({});
23246
23247             e.set(this.parent.displayField, this.parent.emptyTitle);
23248             e.set(this.parent.valueField, '');
23249
23250             this.insert(0, e);
23251         }
23252             
23253         this.fireEvent("load", this, r, options, o);
23254         if(options.callback){
23255             options.callback.call(options.scope || this, r, options, true);
23256         }
23257     },
23258
23259
23260     /**
23261      * Loads data from a passed data block. A Reader which understands the format of the data
23262      * must have been configured in the constructor.
23263      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23264      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23265      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23266      */
23267     loadData : function(o, append){
23268         var r = this.reader.readRecords(o);
23269         this.loadRecords(r, {add: append}, true);
23270     },
23271
23272     /**
23273      * Gets the number of cached records.
23274      * <p>
23275      * <em>If using paging, this may not be the total size of the dataset. If the data object
23276      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23277      * the data set size</em>
23278      */
23279     getCount : function(){
23280         return this.data.length || 0;
23281     },
23282
23283     /**
23284      * Gets the total number of records in the dataset as returned by the server.
23285      * <p>
23286      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23287      * the dataset size</em>
23288      */
23289     getTotalCount : function(){
23290         return this.totalLength || 0;
23291     },
23292
23293     /**
23294      * Returns the sort state of the Store as an object with two properties:
23295      * <pre><code>
23296  field {String} The name of the field by which the Records are sorted
23297  direction {String} The sort order, "ASC" or "DESC"
23298      * </code></pre>
23299      */
23300     getSortState : function(){
23301         return this.sortInfo;
23302     },
23303
23304     // private
23305     applySort : function(){
23306         if(this.sortInfo && !this.remoteSort){
23307             var s = this.sortInfo, f = s.field;
23308             var st = this.fields.get(f).sortType;
23309             var fn = function(r1, r2){
23310                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23311                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23312             };
23313             this.data.sort(s.direction, fn);
23314             if(this.snapshot && this.snapshot != this.data){
23315                 this.snapshot.sort(s.direction, fn);
23316             }
23317         }
23318     },
23319
23320     /**
23321      * Sets the default sort column and order to be used by the next load operation.
23322      * @param {String} fieldName The name of the field to sort by.
23323      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23324      */
23325     setDefaultSort : function(field, dir){
23326         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23327     },
23328
23329     /**
23330      * Sort the Records.
23331      * If remote sorting is used, the sort is performed on the server, and the cache is
23332      * reloaded. If local sorting is used, the cache is sorted internally.
23333      * @param {String} fieldName The name of the field to sort by.
23334      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23335      */
23336     sort : function(fieldName, dir){
23337         var f = this.fields.get(fieldName);
23338         if(!dir){
23339             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23340             
23341             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23342                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23343             }else{
23344                 dir = f.sortDir;
23345             }
23346         }
23347         this.sortToggle[f.name] = dir;
23348         this.sortInfo = {field: f.name, direction: dir};
23349         if(!this.remoteSort){
23350             this.applySort();
23351             this.fireEvent("datachanged", this);
23352         }else{
23353             this.load(this.lastOptions);
23354         }
23355     },
23356
23357     /**
23358      * Calls the specified function for each of the Records in the cache.
23359      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23360      * Returning <em>false</em> aborts and exits the iteration.
23361      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23362      */
23363     each : function(fn, scope){
23364         this.data.each(fn, scope);
23365     },
23366
23367     /**
23368      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23369      * (e.g., during paging).
23370      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23371      */
23372     getModifiedRecords : function(){
23373         return this.modified;
23374     },
23375
23376     // private
23377     createFilterFn : function(property, value, anyMatch){
23378         if(!value.exec){ // not a regex
23379             value = String(value);
23380             if(value.length == 0){
23381                 return false;
23382             }
23383             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23384         }
23385         return function(r){
23386             return value.test(r.data[property]);
23387         };
23388     },
23389
23390     /**
23391      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23392      * @param {String} property A field on your records
23393      * @param {Number} start The record index to start at (defaults to 0)
23394      * @param {Number} end The last record index to include (defaults to length - 1)
23395      * @return {Number} The sum
23396      */
23397     sum : function(property, start, end){
23398         var rs = this.data.items, v = 0;
23399         start = start || 0;
23400         end = (end || end === 0) ? end : rs.length-1;
23401
23402         for(var i = start; i <= end; i++){
23403             v += (rs[i].data[property] || 0);
23404         }
23405         return v;
23406     },
23407
23408     /**
23409      * Filter the records by a specified property.
23410      * @param {String} field A field on your records
23411      * @param {String/RegExp} value Either a string that the field
23412      * should start with or a RegExp to test against the field
23413      * @param {Boolean} anyMatch True to match any part not just the beginning
23414      */
23415     filter : function(property, value, anyMatch){
23416         var fn = this.createFilterFn(property, value, anyMatch);
23417         return fn ? this.filterBy(fn) : this.clearFilter();
23418     },
23419
23420     /**
23421      * Filter by a function. The specified function will be called with each
23422      * record in this data source. If the function returns true the record is included,
23423      * otherwise it is filtered.
23424      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23425      * @param {Object} scope (optional) The scope of the function (defaults to this)
23426      */
23427     filterBy : function(fn, scope){
23428         this.snapshot = this.snapshot || this.data;
23429         this.data = this.queryBy(fn, scope||this);
23430         this.fireEvent("datachanged", this);
23431     },
23432
23433     /**
23434      * Query the records by a specified property.
23435      * @param {String} field A field on your records
23436      * @param {String/RegExp} value Either a string that the field
23437      * should start with or a RegExp to test against the field
23438      * @param {Boolean} anyMatch True to match any part not just the beginning
23439      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23440      */
23441     query : function(property, value, anyMatch){
23442         var fn = this.createFilterFn(property, value, anyMatch);
23443         return fn ? this.queryBy(fn) : this.data.clone();
23444     },
23445
23446     /**
23447      * Query by a function. The specified function will be called with each
23448      * record in this data source. If the function returns true the record is included
23449      * in the results.
23450      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23451      * @param {Object} scope (optional) The scope of the function (defaults to this)
23452       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23453      **/
23454     queryBy : function(fn, scope){
23455         var data = this.snapshot || this.data;
23456         return data.filterBy(fn, scope||this);
23457     },
23458
23459     /**
23460      * Collects unique values for a particular dataIndex from this store.
23461      * @param {String} dataIndex The property to collect
23462      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23463      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23464      * @return {Array} An array of the unique values
23465      **/
23466     collect : function(dataIndex, allowNull, bypassFilter){
23467         var d = (bypassFilter === true && this.snapshot) ?
23468                 this.snapshot.items : this.data.items;
23469         var v, sv, r = [], l = {};
23470         for(var i = 0, len = d.length; i < len; i++){
23471             v = d[i].data[dataIndex];
23472             sv = String(v);
23473             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23474                 l[sv] = true;
23475                 r[r.length] = v;
23476             }
23477         }
23478         return r;
23479     },
23480
23481     /**
23482      * Revert to a view of the Record cache with no filtering applied.
23483      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23484      */
23485     clearFilter : function(suppressEvent){
23486         if(this.snapshot && this.snapshot != this.data){
23487             this.data = this.snapshot;
23488             delete this.snapshot;
23489             if(suppressEvent !== true){
23490                 this.fireEvent("datachanged", this);
23491             }
23492         }
23493     },
23494
23495     // private
23496     afterEdit : function(record){
23497         if(this.modified.indexOf(record) == -1){
23498             this.modified.push(record);
23499         }
23500         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23501     },
23502     
23503     // private
23504     afterReject : function(record){
23505         this.modified.remove(record);
23506         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23507     },
23508
23509     // private
23510     afterCommit : function(record){
23511         this.modified.remove(record);
23512         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23513     },
23514
23515     /**
23516      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23517      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23518      */
23519     commitChanges : function(){
23520         var m = this.modified.slice(0);
23521         this.modified = [];
23522         for(var i = 0, len = m.length; i < len; i++){
23523             m[i].commit();
23524         }
23525     },
23526
23527     /**
23528      * Cancel outstanding changes on all changed records.
23529      */
23530     rejectChanges : function(){
23531         var m = this.modified.slice(0);
23532         this.modified = [];
23533         for(var i = 0, len = m.length; i < len; i++){
23534             m[i].reject();
23535         }
23536     },
23537
23538     onMetaChange : function(meta, rtype, o){
23539         this.recordType = rtype;
23540         this.fields = rtype.prototype.fields;
23541         delete this.snapshot;
23542         this.sortInfo = meta.sortInfo || this.sortInfo;
23543         this.modified = [];
23544         this.fireEvent('metachange', this, this.reader.meta);
23545     },
23546     
23547     moveIndex : function(data, type)
23548     {
23549         var index = this.indexOf(data);
23550         
23551         var newIndex = index + type;
23552         
23553         this.remove(data);
23554         
23555         this.insert(newIndex, data);
23556         
23557     }
23558 });/*
23559  * Based on:
23560  * Ext JS Library 1.1.1
23561  * Copyright(c) 2006-2007, Ext JS, LLC.
23562  *
23563  * Originally Released Under LGPL - original licence link has changed is not relivant.
23564  *
23565  * Fork - LGPL
23566  * <script type="text/javascript">
23567  */
23568
23569 /**
23570  * @class Roo.data.SimpleStore
23571  * @extends Roo.data.Store
23572  * Small helper class to make creating Stores from Array data easier.
23573  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23574  * @cfg {Array} fields An array of field definition objects, or field name strings.
23575  * @cfg {Object} an existing reader (eg. copied from another store)
23576  * @cfg {Array} data The multi-dimensional array of data
23577  * @constructor
23578  * @param {Object} config
23579  */
23580 Roo.data.SimpleStore = function(config)
23581 {
23582     Roo.data.SimpleStore.superclass.constructor.call(this, {
23583         isLocal : true,
23584         reader: typeof(config.reader) != 'undefined' ? config.reader : new Roo.data.ArrayReader({
23585                 id: config.id
23586             },
23587             Roo.data.Record.create(config.fields)
23588         ),
23589         proxy : new Roo.data.MemoryProxy(config.data)
23590     });
23591     this.load();
23592 };
23593 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23594  * Based on:
23595  * Ext JS Library 1.1.1
23596  * Copyright(c) 2006-2007, Ext JS, LLC.
23597  *
23598  * Originally Released Under LGPL - original licence link has changed is not relivant.
23599  *
23600  * Fork - LGPL
23601  * <script type="text/javascript">
23602  */
23603
23604 /**
23605 /**
23606  * @extends Roo.data.Store
23607  * @class Roo.data.JsonStore
23608  * Small helper class to make creating Stores for JSON data easier. <br/>
23609 <pre><code>
23610 var store = new Roo.data.JsonStore({
23611     url: 'get-images.php',
23612     root: 'images',
23613     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23614 });
23615 </code></pre>
23616  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23617  * JsonReader and HttpProxy (unless inline data is provided).</b>
23618  * @cfg {Array} fields An array of field definition objects, or field name strings.
23619  * @constructor
23620  * @param {Object} config
23621  */
23622 Roo.data.JsonStore = function(c){
23623     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23624         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23625         reader: new Roo.data.JsonReader(c, c.fields)
23626     }));
23627 };
23628 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23629  * Based on:
23630  * Ext JS Library 1.1.1
23631  * Copyright(c) 2006-2007, Ext JS, LLC.
23632  *
23633  * Originally Released Under LGPL - original licence link has changed is not relivant.
23634  *
23635  * Fork - LGPL
23636  * <script type="text/javascript">
23637  */
23638
23639  
23640 Roo.data.Field = function(config){
23641     if(typeof config == "string"){
23642         config = {name: config};
23643     }
23644     Roo.apply(this, config);
23645     
23646     if(!this.type){
23647         this.type = "auto";
23648     }
23649     
23650     var st = Roo.data.SortTypes;
23651     // named sortTypes are supported, here we look them up
23652     if(typeof this.sortType == "string"){
23653         this.sortType = st[this.sortType];
23654     }
23655     
23656     // set default sortType for strings and dates
23657     if(!this.sortType){
23658         switch(this.type){
23659             case "string":
23660                 this.sortType = st.asUCString;
23661                 break;
23662             case "date":
23663                 this.sortType = st.asDate;
23664                 break;
23665             default:
23666                 this.sortType = st.none;
23667         }
23668     }
23669
23670     // define once
23671     var stripRe = /[\$,%]/g;
23672
23673     // prebuilt conversion function for this field, instead of
23674     // switching every time we're reading a value
23675     if(!this.convert){
23676         var cv, dateFormat = this.dateFormat;
23677         switch(this.type){
23678             case "":
23679             case "auto":
23680             case undefined:
23681                 cv = function(v){ return v; };
23682                 break;
23683             case "string":
23684                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23685                 break;
23686             case "int":
23687                 cv = function(v){
23688                     return v !== undefined && v !== null && v !== '' ?
23689                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23690                     };
23691                 break;
23692             case "float":
23693                 cv = function(v){
23694                     return v !== undefined && v !== null && v !== '' ?
23695                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23696                     };
23697                 break;
23698             case "bool":
23699             case "boolean":
23700                 cv = function(v){ return v === true || v === "true" || v == 1; };
23701                 break;
23702             case "date":
23703                 cv = function(v){
23704                     if(!v){
23705                         return '';
23706                     }
23707                     if(v instanceof Date){
23708                         return v;
23709                     }
23710                     if(dateFormat){
23711                         if(dateFormat == "timestamp"){
23712                             return new Date(v*1000);
23713                         }
23714                         return Date.parseDate(v, dateFormat);
23715                     }
23716                     var parsed = Date.parse(v);
23717                     return parsed ? new Date(parsed) : null;
23718                 };
23719              break;
23720             
23721         }
23722         this.convert = cv;
23723     }
23724 };
23725
23726 Roo.data.Field.prototype = {
23727     dateFormat: null,
23728     defaultValue: "",
23729     mapping: null,
23730     sortType : null,
23731     sortDir : "ASC"
23732 };/*
23733  * Based on:
23734  * Ext JS Library 1.1.1
23735  * Copyright(c) 2006-2007, Ext JS, LLC.
23736  *
23737  * Originally Released Under LGPL - original licence link has changed is not relivant.
23738  *
23739  * Fork - LGPL
23740  * <script type="text/javascript">
23741  */
23742  
23743 // Base class for reading structured data from a data source.  This class is intended to be
23744 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23745
23746 /**
23747  * @class Roo.data.DataReader
23748  * Base class for reading structured data from a data source.  This class is intended to be
23749  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23750  */
23751
23752 Roo.data.DataReader = function(meta, recordType){
23753     
23754     this.meta = meta;
23755     
23756     this.recordType = recordType instanceof Array ? 
23757         Roo.data.Record.create(recordType) : recordType;
23758 };
23759
23760 Roo.data.DataReader.prototype = {
23761     
23762     
23763     readerType : 'Data',
23764      /**
23765      * Create an empty record
23766      * @param {Object} data (optional) - overlay some values
23767      * @return {Roo.data.Record} record created.
23768      */
23769     newRow :  function(d) {
23770         var da =  {};
23771         this.recordType.prototype.fields.each(function(c) {
23772             switch( c.type) {
23773                 case 'int' : da[c.name] = 0; break;
23774                 case 'date' : da[c.name] = new Date(); break;
23775                 case 'float' : da[c.name] = 0.0; break;
23776                 case 'boolean' : da[c.name] = false; break;
23777                 default : da[c.name] = ""; break;
23778             }
23779             
23780         });
23781         return new this.recordType(Roo.apply(da, d));
23782     }
23783     
23784     
23785 };/*
23786  * Based on:
23787  * Ext JS Library 1.1.1
23788  * Copyright(c) 2006-2007, Ext JS, LLC.
23789  *
23790  * Originally Released Under LGPL - original licence link has changed is not relivant.
23791  *
23792  * Fork - LGPL
23793  * <script type="text/javascript">
23794  */
23795
23796 /**
23797  * @class Roo.data.DataProxy
23798  * @extends Roo.data.Observable
23799  * This class is an abstract base class for implementations which provide retrieval of
23800  * unformatted data objects.<br>
23801  * <p>
23802  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23803  * (of the appropriate type which knows how to parse the data object) to provide a block of
23804  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23805  * <p>
23806  * Custom implementations must implement the load method as described in
23807  * {@link Roo.data.HttpProxy#load}.
23808  */
23809 Roo.data.DataProxy = function(){
23810     this.addEvents({
23811         /**
23812          * @event beforeload
23813          * Fires before a network request is made to retrieve a data object.
23814          * @param {Object} This DataProxy object.
23815          * @param {Object} params The params parameter to the load function.
23816          */
23817         beforeload : true,
23818         /**
23819          * @event load
23820          * Fires before the load method's callback is called.
23821          * @param {Object} This DataProxy object.
23822          * @param {Object} o The data object.
23823          * @param {Object} arg The callback argument object passed to the load function.
23824          */
23825         load : true,
23826         /**
23827          * @event loadexception
23828          * Fires if an Exception occurs during data retrieval.
23829          * @param {Object} This DataProxy object.
23830          * @param {Object} o The data object.
23831          * @param {Object} arg The callback argument object passed to the load function.
23832          * @param {Object} e The Exception.
23833          */
23834         loadexception : true
23835     });
23836     Roo.data.DataProxy.superclass.constructor.call(this);
23837 };
23838
23839 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23840
23841     /**
23842      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23843      */
23844 /*
23845  * Based on:
23846  * Ext JS Library 1.1.1
23847  * Copyright(c) 2006-2007, Ext JS, LLC.
23848  *
23849  * Originally Released Under LGPL - original licence link has changed is not relivant.
23850  *
23851  * Fork - LGPL
23852  * <script type="text/javascript">
23853  */
23854 /**
23855  * @class Roo.data.MemoryProxy
23856  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23857  * to the Reader when its load method is called.
23858  * @constructor
23859  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23860  */
23861 Roo.data.MemoryProxy = function(data){
23862     if (data.data) {
23863         data = data.data;
23864     }
23865     Roo.data.MemoryProxy.superclass.constructor.call(this);
23866     this.data = data;
23867 };
23868
23869 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23870     
23871     /**
23872      * Load data from the requested source (in this case an in-memory
23873      * data object passed to the constructor), read the data object into
23874      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23875      * process that block using the passed callback.
23876      * @param {Object} params This parameter is not used by the MemoryProxy class.
23877      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23878      * object into a block of Roo.data.Records.
23879      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23880      * The function must be passed <ul>
23881      * <li>The Record block object</li>
23882      * <li>The "arg" argument from the load function</li>
23883      * <li>A boolean success indicator</li>
23884      * </ul>
23885      * @param {Object} scope The scope in which to call the callback
23886      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23887      */
23888     load : function(params, reader, callback, scope, arg){
23889         params = params || {};
23890         var result;
23891         try {
23892             result = reader.readRecords(params.data ? params.data :this.data);
23893         }catch(e){
23894             this.fireEvent("loadexception", this, arg, null, e);
23895             callback.call(scope, null, arg, false);
23896             return;
23897         }
23898         callback.call(scope, result, arg, true);
23899     },
23900     
23901     // private
23902     update : function(params, records){
23903         
23904     }
23905 });/*
23906  * Based on:
23907  * Ext JS Library 1.1.1
23908  * Copyright(c) 2006-2007, Ext JS, LLC.
23909  *
23910  * Originally Released Under LGPL - original licence link has changed is not relivant.
23911  *
23912  * Fork - LGPL
23913  * <script type="text/javascript">
23914  */
23915 /**
23916  * @class Roo.data.HttpProxy
23917  * @extends Roo.data.DataProxy
23918  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23919  * configured to reference a certain URL.<br><br>
23920  * <p>
23921  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23922  * from which the running page was served.<br><br>
23923  * <p>
23924  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23925  * <p>
23926  * Be aware that to enable the browser to parse an XML document, the server must set
23927  * the Content-Type header in the HTTP response to "text/xml".
23928  * @constructor
23929  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23930  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23931  * will be used to make the request.
23932  */
23933 Roo.data.HttpProxy = function(conn){
23934     Roo.data.HttpProxy.superclass.constructor.call(this);
23935     // is conn a conn config or a real conn?
23936     this.conn = conn;
23937     this.useAjax = !conn || !conn.events;
23938   
23939 };
23940
23941 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23942     // thse are take from connection...
23943     
23944     /**
23945      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23946      */
23947     /**
23948      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23949      * extra parameters to each request made by this object. (defaults to undefined)
23950      */
23951     /**
23952      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23953      *  to each request made by this object. (defaults to undefined)
23954      */
23955     /**
23956      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
23957      */
23958     /**
23959      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23960      */
23961      /**
23962      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23963      * @type Boolean
23964      */
23965   
23966
23967     /**
23968      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23969      * @type Boolean
23970      */
23971     /**
23972      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23973      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23974      * a finer-grained basis than the DataProxy events.
23975      */
23976     getConnection : function(){
23977         return this.useAjax ? Roo.Ajax : this.conn;
23978     },
23979
23980     /**
23981      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23982      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23983      * process that block using the passed callback.
23984      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23985      * for the request to the remote server.
23986      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23987      * object into a block of Roo.data.Records.
23988      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23989      * The function must be passed <ul>
23990      * <li>The Record block object</li>
23991      * <li>The "arg" argument from the load function</li>
23992      * <li>A boolean success indicator</li>
23993      * </ul>
23994      * @param {Object} scope The scope in which to call the callback
23995      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23996      */
23997     load : function(params, reader, callback, scope, arg){
23998         if(this.fireEvent("beforeload", this, params) !== false){
23999             var  o = {
24000                 params : params || {},
24001                 request: {
24002                     callback : callback,
24003                     scope : scope,
24004                     arg : arg
24005                 },
24006                 reader: reader,
24007                 callback : this.loadResponse,
24008                 scope: this
24009             };
24010             if(this.useAjax){
24011                 Roo.applyIf(o, this.conn);
24012                 if(this.activeRequest){
24013                     Roo.Ajax.abort(this.activeRequest);
24014                 }
24015                 this.activeRequest = Roo.Ajax.request(o);
24016             }else{
24017                 this.conn.request(o);
24018             }
24019         }else{
24020             callback.call(scope||this, null, arg, false);
24021         }
24022     },
24023
24024     // private
24025     loadResponse : function(o, success, response){
24026         delete this.activeRequest;
24027         if(!success){
24028             this.fireEvent("loadexception", this, o, response);
24029             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24030             return;
24031         }
24032         var result;
24033         try {
24034             result = o.reader.read(response);
24035         }catch(e){
24036             this.fireEvent("loadexception", this, o, response, e);
24037             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24038             return;
24039         }
24040         
24041         this.fireEvent("load", this, o, o.request.arg);
24042         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24043     },
24044
24045     // private
24046     update : function(dataSet){
24047
24048     },
24049
24050     // private
24051     updateResponse : function(dataSet){
24052
24053     }
24054 });/*
24055  * Based on:
24056  * Ext JS Library 1.1.1
24057  * Copyright(c) 2006-2007, Ext JS, LLC.
24058  *
24059  * Originally Released Under LGPL - original licence link has changed is not relivant.
24060  *
24061  * Fork - LGPL
24062  * <script type="text/javascript">
24063  */
24064
24065 /**
24066  * @class Roo.data.ScriptTagProxy
24067  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24068  * other than the originating domain of the running page.<br><br>
24069  * <p>
24070  * <em>Note that if you are retrieving data from a page that is in a domain that is NOT the same as the originating domain
24071  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24072  * <p>
24073  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24074  * source code that is used as the source inside a &lt;script> tag.<br><br>
24075  * <p>
24076  * In order for the browser to process the returned data, the server must wrap the data object
24077  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24078  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24079  * depending on whether the callback name was passed:
24080  * <p>
24081  * <pre><code>
24082 boolean scriptTag = false;
24083 String cb = request.getParameter("callback");
24084 if (cb != null) {
24085     scriptTag = true;
24086     response.setContentType("text/javascript");
24087 } else {
24088     response.setContentType("application/x-json");
24089 }
24090 Writer out = response.getWriter();
24091 if (scriptTag) {
24092     out.write(cb + "(");
24093 }
24094 out.print(dataBlock.toJsonString());
24095 if (scriptTag) {
24096     out.write(");");
24097 }
24098 </pre></code>
24099  *
24100  * @constructor
24101  * @param {Object} config A configuration object.
24102  */
24103 Roo.data.ScriptTagProxy = function(config){
24104     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24105     Roo.apply(this, config);
24106     this.head = document.getElementsByTagName("head")[0];
24107 };
24108
24109 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24110
24111 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24112     /**
24113      * @cfg {String} url The URL from which to request the data object.
24114      */
24115     /**
24116      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24117      */
24118     timeout : 30000,
24119     /**
24120      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24121      * the server the name of the callback function set up by the load call to process the returned data object.
24122      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24123      * javascript output which calls this named function passing the data object as its only parameter.
24124      */
24125     callbackParam : "callback",
24126     /**
24127      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24128      * name to the request.
24129      */
24130     nocache : true,
24131
24132     /**
24133      * Load data from the configured URL, read the data object into
24134      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24135      * process that block using the passed callback.
24136      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24137      * for the request to the remote server.
24138      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24139      * object into a block of Roo.data.Records.
24140      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24141      * The function must be passed <ul>
24142      * <li>The Record block object</li>
24143      * <li>The "arg" argument from the load function</li>
24144      * <li>A boolean success indicator</li>
24145      * </ul>
24146      * @param {Object} scope The scope in which to call the callback
24147      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24148      */
24149     load : function(params, reader, callback, scope, arg){
24150         if(this.fireEvent("beforeload", this, params) !== false){
24151
24152             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24153
24154             var url = this.url;
24155             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24156             if(this.nocache){
24157                 url += "&_dc=" + (new Date().getTime());
24158             }
24159             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24160             var trans = {
24161                 id : transId,
24162                 cb : "stcCallback"+transId,
24163                 scriptId : "stcScript"+transId,
24164                 params : params,
24165                 arg : arg,
24166                 url : url,
24167                 callback : callback,
24168                 scope : scope,
24169                 reader : reader
24170             };
24171             var conn = this;
24172
24173             window[trans.cb] = function(o){
24174                 conn.handleResponse(o, trans);
24175             };
24176
24177             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24178
24179             if(this.autoAbort !== false){
24180                 this.abort();
24181             }
24182
24183             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24184
24185             var script = document.createElement("script");
24186             script.setAttribute("src", url);
24187             script.setAttribute("type", "text/javascript");
24188             script.setAttribute("id", trans.scriptId);
24189             this.head.appendChild(script);
24190
24191             this.trans = trans;
24192         }else{
24193             callback.call(scope||this, null, arg, false);
24194         }
24195     },
24196
24197     // private
24198     isLoading : function(){
24199         return this.trans ? true : false;
24200     },
24201
24202     /**
24203      * Abort the current server request.
24204      */
24205     abort : function(){
24206         if(this.isLoading()){
24207             this.destroyTrans(this.trans);
24208         }
24209     },
24210
24211     // private
24212     destroyTrans : function(trans, isLoaded){
24213         this.head.removeChild(document.getElementById(trans.scriptId));
24214         clearTimeout(trans.timeoutId);
24215         if(isLoaded){
24216             window[trans.cb] = undefined;
24217             try{
24218                 delete window[trans.cb];
24219             }catch(e){}
24220         }else{
24221             // if hasn't been loaded, wait for load to remove it to prevent script error
24222             window[trans.cb] = function(){
24223                 window[trans.cb] = undefined;
24224                 try{
24225                     delete window[trans.cb];
24226                 }catch(e){}
24227             };
24228         }
24229     },
24230
24231     // private
24232     handleResponse : function(o, trans){
24233         this.trans = false;
24234         this.destroyTrans(trans, true);
24235         var result;
24236         try {
24237             result = trans.reader.readRecords(o);
24238         }catch(e){
24239             this.fireEvent("loadexception", this, o, trans.arg, e);
24240             trans.callback.call(trans.scope||window, null, trans.arg, false);
24241             return;
24242         }
24243         this.fireEvent("load", this, o, trans.arg);
24244         trans.callback.call(trans.scope||window, result, trans.arg, true);
24245     },
24246
24247     // private
24248     handleFailure : function(trans){
24249         this.trans = false;
24250         this.destroyTrans(trans, false);
24251         this.fireEvent("loadexception", this, null, trans.arg);
24252         trans.callback.call(trans.scope||window, null, trans.arg, false);
24253     }
24254 });/*
24255  * Based on:
24256  * Ext JS Library 1.1.1
24257  * Copyright(c) 2006-2007, Ext JS, LLC.
24258  *
24259  * Originally Released Under LGPL - original licence link has changed is not relivant.
24260  *
24261  * Fork - LGPL
24262  * <script type="text/javascript">
24263  */
24264
24265 /**
24266  * @class Roo.data.JsonReader
24267  * @extends Roo.data.DataReader
24268  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24269  * based on mappings in a provided Roo.data.Record constructor.
24270  * 
24271  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24272  * in the reply previously. 
24273  * 
24274  * <p>
24275  * Example code:
24276  * <pre><code>
24277 var RecordDef = Roo.data.Record.create([
24278     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24279     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24280 ]);
24281 var myReader = new Roo.data.JsonReader({
24282     totalProperty: "results",    // The property which contains the total dataset size (optional)
24283     root: "rows",                // The property which contains an Array of row objects
24284     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24285 }, RecordDef);
24286 </code></pre>
24287  * <p>
24288  * This would consume a JSON file like this:
24289  * <pre><code>
24290 { 'results': 2, 'rows': [
24291     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24292     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24293 }
24294 </code></pre>
24295  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24296  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24297  * paged from the remote server.
24298  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24299  * @cfg {String} root name of the property which contains the Array of row objects.
24300  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24301  * @cfg {Array} fields Array of field definition objects
24302  * @constructor
24303  * Create a new JsonReader
24304  * @param {Object} meta Metadata configuration options
24305  * @param {Object} recordType Either an Array of field definition objects,
24306  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24307  */
24308 Roo.data.JsonReader = function(meta, recordType){
24309     
24310     meta = meta || {};
24311     // set some defaults:
24312     Roo.applyIf(meta, {
24313         totalProperty: 'total',
24314         successProperty : 'success',
24315         root : 'data',
24316         id : 'id'
24317     });
24318     
24319     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24320 };
24321 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24322     
24323     readerType : 'Json',
24324     
24325     /**
24326      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24327      * Used by Store query builder to append _requestMeta to params.
24328      * 
24329      */
24330     metaFromRemote : false,
24331     /**
24332      * This method is only used by a DataProxy which has retrieved data from a remote server.
24333      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24334      * @return {Object} data A data block which is used by an Roo.data.Store object as
24335      * a cache of Roo.data.Records.
24336      */
24337     read : function(response){
24338         var json = response.responseText;
24339        
24340         var o = /* eval:var:o */ eval("("+json+")");
24341         if(!o) {
24342             throw {message: "JsonReader.read: Json object not found"};
24343         }
24344         
24345         if(o.metaData){
24346             
24347             delete this.ef;
24348             this.metaFromRemote = true;
24349             this.meta = o.metaData;
24350             this.recordType = Roo.data.Record.create(o.metaData.fields);
24351             this.onMetaChange(this.meta, this.recordType, o);
24352         }
24353         return this.readRecords(o);
24354     },
24355
24356     // private function a store will implement
24357     onMetaChange : function(meta, recordType, o){
24358
24359     },
24360
24361     /**
24362          * @ignore
24363          */
24364     simpleAccess: function(obj, subsc) {
24365         return obj[subsc];
24366     },
24367
24368         /**
24369          * @ignore
24370          */
24371     getJsonAccessor: function(){
24372         var re = /[\[\.]/;
24373         return function(expr) {
24374             try {
24375                 return(re.test(expr))
24376                     ? new Function("obj", "return obj." + expr)
24377                     : function(obj){
24378                         return obj[expr];
24379                     };
24380             } catch(e){}
24381             return Roo.emptyFn;
24382         };
24383     }(),
24384
24385     /**
24386      * Create a data block containing Roo.data.Records from an XML document.
24387      * @param {Object} o An object which contains an Array of row objects in the property specified
24388      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24389      * which contains the total size of the dataset.
24390      * @return {Object} data A data block which is used by an Roo.data.Store object as
24391      * a cache of Roo.data.Records.
24392      */
24393     readRecords : function(o){
24394         /**
24395          * After any data loads, the raw JSON data is available for further custom processing.
24396          * @type Object
24397          */
24398         this.o = o;
24399         var s = this.meta, Record = this.recordType,
24400             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24401
24402 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24403         if (!this.ef) {
24404             if(s.totalProperty) {
24405                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24406                 }
24407                 if(s.successProperty) {
24408                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24409                 }
24410                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24411                 if (s.id) {
24412                         var g = this.getJsonAccessor(s.id);
24413                         this.getId = function(rec) {
24414                                 var r = g(rec);  
24415                                 return (r === undefined || r === "") ? null : r;
24416                         };
24417                 } else {
24418                         this.getId = function(){return null;};
24419                 }
24420             this.ef = [];
24421             for(var jj = 0; jj < fl; jj++){
24422                 f = fi[jj];
24423                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24424                 this.ef[jj] = this.getJsonAccessor(map);
24425             }
24426         }
24427
24428         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24429         if(s.totalProperty){
24430             var vt = parseInt(this.getTotal(o), 10);
24431             if(!isNaN(vt)){
24432                 totalRecords = vt;
24433             }
24434         }
24435         if(s.successProperty){
24436             var vs = this.getSuccess(o);
24437             if(vs === false || vs === 'false'){
24438                 success = false;
24439             }
24440         }
24441         var records = [];
24442         for(var i = 0; i < c; i++){
24443                 var n = root[i];
24444             var values = {};
24445             var id = this.getId(n);
24446             for(var j = 0; j < fl; j++){
24447                 f = fi[j];
24448             var v = this.ef[j](n);
24449             if (!f.convert) {
24450                 Roo.log('missing convert for ' + f.name);
24451                 Roo.log(f);
24452                 continue;
24453             }
24454             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24455             }
24456             var record = new Record(values, id);
24457             record.json = n;
24458             records[i] = record;
24459         }
24460         return {
24461             raw : o,
24462             success : success,
24463             records : records,
24464             totalRecords : totalRecords
24465         };
24466     },
24467     /**
24468      * using 'cn' the nested child reader read the child array into it's child stores.
24469      * @param {Object} rec The record with a 'children array
24470      */
24471     loadDataFromChildren: function(rec)
24472     {
24473         // expect rec just to be an array.. eg [a,b,c, [...] << cn ]
24474         var data = typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
24475         return this.loadData({ data : data, total : data.length });
24476         
24477     }
24478 });/*
24479  * Based on:
24480  * Ext JS Library 1.1.1
24481  * Copyright(c) 2006-2007, Ext JS, LLC.
24482  *
24483  * Originally Released Under LGPL - original licence link has changed is not relivant.
24484  *
24485  * Fork - LGPL
24486  * <script type="text/javascript">
24487  */
24488
24489 /**
24490  * @class Roo.data.XmlReader
24491  * @extends Roo.data.DataReader
24492  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24493  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24494  * <p>
24495  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24496  * header in the HTTP response must be set to "text/xml".</em>
24497  * <p>
24498  * Example code:
24499  * <pre><code>
24500 var RecordDef = Roo.data.Record.create([
24501    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24502    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24503 ]);
24504 var myReader = new Roo.data.XmlReader({
24505    totalRecords: "results", // The element which contains the total dataset size (optional)
24506    record: "row",           // The repeated element which contains row information
24507    id: "id"                 // The element within the row that provides an ID for the record (optional)
24508 }, RecordDef);
24509 </code></pre>
24510  * <p>
24511  * This would consume an XML file like this:
24512  * <pre><code>
24513 &lt;?xml?>
24514 &lt;dataset>
24515  &lt;results>2&lt;/results>
24516  &lt;row>
24517    &lt;id>1&lt;/id>
24518    &lt;name>Bill&lt;/name>
24519    &lt;occupation>Gardener&lt;/occupation>
24520  &lt;/row>
24521  &lt;row>
24522    &lt;id>2&lt;/id>
24523    &lt;name>Ben&lt;/name>
24524    &lt;occupation>Horticulturalist&lt;/occupation>
24525  &lt;/row>
24526 &lt;/dataset>
24527 </code></pre>
24528  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24529  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24530  * paged from the remote server.
24531  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24532  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24533  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24534  * a record identifier value.
24535  * @constructor
24536  * Create a new XmlReader
24537  * @param {Object} meta Metadata configuration options
24538  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24539  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24540  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24541  */
24542 Roo.data.XmlReader = function(meta, recordType){
24543     meta = meta || {};
24544     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24545 };
24546 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24547     
24548     readerType : 'Xml',
24549     
24550     /**
24551      * This method is only used by a DataProxy which has retrieved data from a remote server.
24552          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24553          * to contain a method called 'responseXML' that returns an XML document object.
24554      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24555      * a cache of Roo.data.Records.
24556      */
24557     read : function(response){
24558         var doc = response.responseXML;
24559         if(!doc) {
24560             throw {message: "XmlReader.read: XML Document not available"};
24561         }
24562         return this.readRecords(doc);
24563     },
24564
24565     /**
24566      * Create a data block containing Roo.data.Records from an XML document.
24567          * @param {Object} doc A parsed XML document.
24568      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24569      * a cache of Roo.data.Records.
24570      */
24571     readRecords : function(doc){
24572         /**
24573          * After any data loads/reads, the raw XML Document is available for further custom processing.
24574          * @type XMLDocument
24575          */
24576         this.xmlData = doc;
24577         var root = doc.documentElement || doc;
24578         var q = Roo.DomQuery;
24579         var recordType = this.recordType, fields = recordType.prototype.fields;
24580         var sid = this.meta.id;
24581         var totalRecords = 0, success = true;
24582         if(this.meta.totalRecords){
24583             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24584         }
24585         
24586         if(this.meta.success){
24587             var sv = q.selectValue(this.meta.success, root, true);
24588             success = sv !== false && sv !== 'false';
24589         }
24590         var records = [];
24591         var ns = q.select(this.meta.record, root);
24592         for(var i = 0, len = ns.length; i < len; i++) {
24593                 var n = ns[i];
24594                 var values = {};
24595                 var id = sid ? q.selectValue(sid, n) : undefined;
24596                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24597                     var f = fields.items[j];
24598                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24599                     v = f.convert(v);
24600                     values[f.name] = v;
24601                 }
24602                 var record = new recordType(values, id);
24603                 record.node = n;
24604                 records[records.length] = record;
24605             }
24606
24607             return {
24608                 success : success,
24609                 records : records,
24610                 totalRecords : totalRecords || records.length
24611             };
24612     }
24613 });/*
24614  * Based on:
24615  * Ext JS Library 1.1.1
24616  * Copyright(c) 2006-2007, Ext JS, LLC.
24617  *
24618  * Originally Released Under LGPL - original licence link has changed is not relivant.
24619  *
24620  * Fork - LGPL
24621  * <script type="text/javascript">
24622  */
24623
24624 /**
24625  * @class Roo.data.ArrayReader
24626  * @extends Roo.data.DataReader
24627  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24628  * Each element of that Array represents a row of data fields. The
24629  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24630  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24631  * <p>
24632  * Example code:.
24633  * <pre><code>
24634 var RecordDef = Roo.data.Record.create([
24635     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24636     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24637 ]);
24638 var myReader = new Roo.data.ArrayReader({
24639     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24640 }, RecordDef);
24641 </code></pre>
24642  * <p>
24643  * This would consume an Array like this:
24644  * <pre><code>
24645 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24646   </code></pre>
24647  
24648  * @constructor
24649  * Create a new JsonReader
24650  * @param {Object} meta Metadata configuration options.
24651  * @param {Object|Array} recordType Either an Array of field definition objects
24652  * 
24653  * @cfg {Array} fields Array of field definition objects
24654  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24655  * as specified to {@link Roo.data.Record#create},
24656  * or an {@link Roo.data.Record} object
24657  *
24658  * 
24659  * created using {@link Roo.data.Record#create}.
24660  */
24661 Roo.data.ArrayReader = function(meta, recordType)
24662 {    
24663     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24664 };
24665
24666 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24667     
24668       /**
24669      * Create a data block containing Roo.data.Records from an XML document.
24670      * @param {Object} o An Array of row objects which represents the dataset.
24671      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24672      * a cache of Roo.data.Records.
24673      */
24674     readRecords : function(o)
24675     {
24676         var sid = this.meta ? this.meta.id : null;
24677         var recordType = this.recordType, fields = recordType.prototype.fields;
24678         var records = [];
24679         var root = o;
24680         for(var i = 0; i < root.length; i++){
24681                 var n = root[i];
24682             var values = {};
24683             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24684             for(var j = 0, jlen = fields.length; j < jlen; j++){
24685                 var f = fields.items[j];
24686                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24687                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24688                 v = f.convert(v);
24689                 values[f.name] = v;
24690             }
24691             var record = new recordType(values, id);
24692             record.json = n;
24693             records[records.length] = record;
24694         }
24695         return {
24696             records : records,
24697             totalRecords : records.length
24698         };
24699     },
24700     /**
24701      * using 'cn' the nested child reader read the child array into it's child stores.
24702      * @param {Object} rec The record with a 'children array
24703      */
24704     loadDataFromChildren: function(rec)
24705     {
24706         // expect rec just to be an array.. eg [a,b,c, [...] << cn ]
24707         return this.loadData(typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn);
24708         
24709     }
24710     
24711     
24712 });/*
24713  * Based on:
24714  * Ext JS Library 1.1.1
24715  * Copyright(c) 2006-2007, Ext JS, LLC.
24716  *
24717  * Originally Released Under LGPL - original licence link has changed is not relivant.
24718  *
24719  * Fork - LGPL
24720  * <script type="text/javascript">
24721  */
24722
24723
24724 /**
24725  * @class Roo.data.Tree
24726  * @extends Roo.util.Observable
24727  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24728  * in the tree have most standard DOM functionality.
24729  * @constructor
24730  * @param {Node} root (optional) The root node
24731  */
24732 Roo.data.Tree = function(root){
24733    this.nodeHash = {};
24734    /**
24735     * The root node for this tree
24736     * @type Node
24737     */
24738    this.root = null;
24739    if(root){
24740        this.setRootNode(root);
24741    }
24742    this.addEvents({
24743        /**
24744         * @event append
24745         * Fires when a new child node is appended to a node in this tree.
24746         * @param {Tree} tree The owner tree
24747         * @param {Node} parent The parent node
24748         * @param {Node} node The newly appended node
24749         * @param {Number} index The index of the newly appended node
24750         */
24751        "append" : true,
24752        /**
24753         * @event remove
24754         * Fires when a child node is removed from a node in this tree.
24755         * @param {Tree} tree The owner tree
24756         * @param {Node} parent The parent node
24757         * @param {Node} node The child node removed
24758         */
24759        "remove" : true,
24760        /**
24761         * @event move
24762         * Fires when a node is moved to a new location in the tree
24763         * @param {Tree} tree The owner tree
24764         * @param {Node} node The node moved
24765         * @param {Node} oldParent The old parent of this node
24766         * @param {Node} newParent The new parent of this node
24767         * @param {Number} index The index it was moved to
24768         */
24769        "move" : true,
24770        /**
24771         * @event insert
24772         * Fires when a new child node is inserted in a node in this tree.
24773         * @param {Tree} tree The owner tree
24774         * @param {Node} parent The parent node
24775         * @param {Node} node The child node inserted
24776         * @param {Node} refNode The child node the node was inserted before
24777         */
24778        "insert" : true,
24779        /**
24780         * @event beforeappend
24781         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24782         * @param {Tree} tree The owner tree
24783         * @param {Node} parent The parent node
24784         * @param {Node} node The child node to be appended
24785         */
24786        "beforeappend" : true,
24787        /**
24788         * @event beforeremove
24789         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24790         * @param {Tree} tree The owner tree
24791         * @param {Node} parent The parent node
24792         * @param {Node} node The child node to be removed
24793         */
24794        "beforeremove" : true,
24795        /**
24796         * @event beforemove
24797         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24798         * @param {Tree} tree The owner tree
24799         * @param {Node} node The node being moved
24800         * @param {Node} oldParent The parent of the node
24801         * @param {Node} newParent The new parent the node is moving to
24802         * @param {Number} index The index it is being moved to
24803         */
24804        "beforemove" : true,
24805        /**
24806         * @event beforeinsert
24807         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24808         * @param {Tree} tree The owner tree
24809         * @param {Node} parent The parent node
24810         * @param {Node} node The child node to be inserted
24811         * @param {Node} refNode The child node the node is being inserted before
24812         */
24813        "beforeinsert" : true
24814    });
24815
24816     Roo.data.Tree.superclass.constructor.call(this);
24817 };
24818
24819 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24820     pathSeparator: "/",
24821
24822     proxyNodeEvent : function(){
24823         return this.fireEvent.apply(this, arguments);
24824     },
24825
24826     /**
24827      * Returns the root node for this tree.
24828      * @return {Node}
24829      */
24830     getRootNode : function(){
24831         return this.root;
24832     },
24833
24834     /**
24835      * Sets the root node for this tree.
24836      * @param {Node} node
24837      * @return {Node}
24838      */
24839     setRootNode : function(node){
24840         this.root = node;
24841         node.ownerTree = this;
24842         node.isRoot = true;
24843         this.registerNode(node);
24844         return node;
24845     },
24846
24847     /**
24848      * Gets a node in this tree by its id.
24849      * @param {String} id
24850      * @return {Node}
24851      */
24852     getNodeById : function(id){
24853         return this.nodeHash[id];
24854     },
24855
24856     registerNode : function(node){
24857         this.nodeHash[node.id] = node;
24858     },
24859
24860     unregisterNode : function(node){
24861         delete this.nodeHash[node.id];
24862     },
24863
24864     toString : function(){
24865         return "[Tree"+(this.id?" "+this.id:"")+"]";
24866     }
24867 });
24868
24869 /**
24870  * @class Roo.data.Node
24871  * @extends Roo.util.Observable
24872  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24873  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24874  * @constructor
24875  * @param {Object} attributes The attributes/config for the node
24876  */
24877 Roo.data.Node = function(attributes){
24878     /**
24879      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24880      * @type {Object}
24881      */
24882     this.attributes = attributes || {};
24883     this.leaf = this.attributes.leaf;
24884     /**
24885      * The node id. @type String
24886      */
24887     this.id = this.attributes.id;
24888     if(!this.id){
24889         this.id = Roo.id(null, "ynode-");
24890         this.attributes.id = this.id;
24891     }
24892      
24893     
24894     /**
24895      * All child nodes of this node. @type Array
24896      */
24897     this.childNodes = [];
24898     if(!this.childNodes.indexOf){ // indexOf is a must
24899         this.childNodes.indexOf = function(o){
24900             for(var i = 0, len = this.length; i < len; i++){
24901                 if(this[i] == o) {
24902                     return i;
24903                 }
24904             }
24905             return -1;
24906         };
24907     }
24908     /**
24909      * The parent node for this node. @type Node
24910      */
24911     this.parentNode = null;
24912     /**
24913      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24914      */
24915     this.firstChild = null;
24916     /**
24917      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24918      */
24919     this.lastChild = null;
24920     /**
24921      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24922      */
24923     this.previousSibling = null;
24924     /**
24925      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24926      */
24927     this.nextSibling = null;
24928
24929     this.addEvents({
24930        /**
24931         * @event append
24932         * Fires when a new child node is appended
24933         * @param {Tree} tree The owner tree
24934         * @param {Node} this This node
24935         * @param {Node} node The newly appended node
24936         * @param {Number} index The index of the newly appended node
24937         */
24938        "append" : true,
24939        /**
24940         * @event remove
24941         * Fires when a child node is removed
24942         * @param {Tree} tree The owner tree
24943         * @param {Node} this This node
24944         * @param {Node} node The removed node
24945         */
24946        "remove" : true,
24947        /**
24948         * @event move
24949         * Fires when this node is moved to a new location in the tree
24950         * @param {Tree} tree The owner tree
24951         * @param {Node} this This node
24952         * @param {Node} oldParent The old parent of this node
24953         * @param {Node} newParent The new parent of this node
24954         * @param {Number} index The index it was moved to
24955         */
24956        "move" : true,
24957        /**
24958         * @event insert
24959         * Fires when a new child node is inserted.
24960         * @param {Tree} tree The owner tree
24961         * @param {Node} this This node
24962         * @param {Node} node The child node inserted
24963         * @param {Node} refNode The child node the node was inserted before
24964         */
24965        "insert" : true,
24966        /**
24967         * @event beforeappend
24968         * Fires before a new child is appended, return false to cancel the append.
24969         * @param {Tree} tree The owner tree
24970         * @param {Node} this This node
24971         * @param {Node} node The child node to be appended
24972         */
24973        "beforeappend" : true,
24974        /**
24975         * @event beforeremove
24976         * Fires before a child is removed, return false to cancel the remove.
24977         * @param {Tree} tree The owner tree
24978         * @param {Node} this This node
24979         * @param {Node} node The child node to be removed
24980         */
24981        "beforeremove" : true,
24982        /**
24983         * @event beforemove
24984         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24985         * @param {Tree} tree The owner tree
24986         * @param {Node} this This node
24987         * @param {Node} oldParent The parent of this node
24988         * @param {Node} newParent The new parent this node is moving to
24989         * @param {Number} index The index it is being moved to
24990         */
24991        "beforemove" : true,
24992        /**
24993         * @event beforeinsert
24994         * Fires before a new child is inserted, return false to cancel the insert.
24995         * @param {Tree} tree The owner tree
24996         * @param {Node} this This node
24997         * @param {Node} node The child node to be inserted
24998         * @param {Node} refNode The child node the node is being inserted before
24999         */
25000        "beforeinsert" : true
25001    });
25002     this.listeners = this.attributes.listeners;
25003     Roo.data.Node.superclass.constructor.call(this);
25004 };
25005
25006 Roo.extend(Roo.data.Node, Roo.util.Observable, {
25007     fireEvent : function(evtName){
25008         // first do standard event for this node
25009         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
25010             return false;
25011         }
25012         // then bubble it up to the tree if the event wasn't cancelled
25013         var ot = this.getOwnerTree();
25014         if(ot){
25015             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
25016                 return false;
25017             }
25018         }
25019         return true;
25020     },
25021
25022     /**
25023      * Returns true if this node is a leaf
25024      * @return {Boolean}
25025      */
25026     isLeaf : function(){
25027         return this.leaf === true;
25028     },
25029
25030     // private
25031     setFirstChild : function(node){
25032         this.firstChild = node;
25033     },
25034
25035     //private
25036     setLastChild : function(node){
25037         this.lastChild = node;
25038     },
25039
25040
25041     /**
25042      * Returns true if this node is the last child of its parent
25043      * @return {Boolean}
25044      */
25045     isLast : function(){
25046        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25047     },
25048
25049     /**
25050      * Returns true if this node is the first child of its parent
25051      * @return {Boolean}
25052      */
25053     isFirst : function(){
25054        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25055     },
25056
25057     hasChildNodes : function(){
25058         return !this.isLeaf() && this.childNodes.length > 0;
25059     },
25060
25061     /**
25062      * Insert node(s) as the last child node of this node.
25063      * @param {Node/Array} node The node or Array of nodes to append
25064      * @return {Node} The appended node if single append, or null if an array was passed
25065      */
25066     appendChild : function(node){
25067         var multi = false;
25068         if(node instanceof Array){
25069             multi = node;
25070         }else if(arguments.length > 1){
25071             multi = arguments;
25072         }
25073         
25074         // if passed an array or multiple args do them one by one
25075         if(multi){
25076             for(var i = 0, len = multi.length; i < len; i++) {
25077                 this.appendChild(multi[i]);
25078             }
25079         }else{
25080             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25081                 return false;
25082             }
25083             var index = this.childNodes.length;
25084             var oldParent = node.parentNode;
25085             // it's a move, make sure we move it cleanly
25086             if(oldParent){
25087                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25088                     return false;
25089                 }
25090                 oldParent.removeChild(node);
25091             }
25092             
25093             index = this.childNodes.length;
25094             if(index == 0){
25095                 this.setFirstChild(node);
25096             }
25097             this.childNodes.push(node);
25098             node.parentNode = this;
25099             var ps = this.childNodes[index-1];
25100             if(ps){
25101                 node.previousSibling = ps;
25102                 ps.nextSibling = node;
25103             }else{
25104                 node.previousSibling = null;
25105             }
25106             node.nextSibling = null;
25107             this.setLastChild(node);
25108             node.setOwnerTree(this.getOwnerTree());
25109             this.fireEvent("append", this.ownerTree, this, node, index);
25110             if(this.ownerTree) {
25111                 this.ownerTree.fireEvent("appendnode", this, node, index);
25112             }
25113             if(oldParent){
25114                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25115             }
25116             return node;
25117         }
25118     },
25119
25120     /**
25121      * Removes a child node from this node.
25122      * @param {Node} node The node to remove
25123      * @return {Node} The removed node
25124      */
25125     removeChild : function(node){
25126         var index = this.childNodes.indexOf(node);
25127         if(index == -1){
25128             return false;
25129         }
25130         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25131             return false;
25132         }
25133
25134         // remove it from childNodes collection
25135         this.childNodes.splice(index, 1);
25136
25137         // update siblings
25138         if(node.previousSibling){
25139             node.previousSibling.nextSibling = node.nextSibling;
25140         }
25141         if(node.nextSibling){
25142             node.nextSibling.previousSibling = node.previousSibling;
25143         }
25144
25145         // update child refs
25146         if(this.firstChild == node){
25147             this.setFirstChild(node.nextSibling);
25148         }
25149         if(this.lastChild == node){
25150             this.setLastChild(node.previousSibling);
25151         }
25152
25153         node.setOwnerTree(null);
25154         // clear any references from the node
25155         node.parentNode = null;
25156         node.previousSibling = null;
25157         node.nextSibling = null;
25158         this.fireEvent("remove", this.ownerTree, this, node);
25159         return node;
25160     },
25161
25162     /**
25163      * Inserts the first node before the second node in this nodes childNodes collection.
25164      * @param {Node} node The node to insert
25165      * @param {Node} refNode The node to insert before (if null the node is appended)
25166      * @return {Node} The inserted node
25167      */
25168     insertBefore : function(node, refNode){
25169         if(!refNode){ // like standard Dom, refNode can be null for append
25170             return this.appendChild(node);
25171         }
25172         // nothing to do
25173         if(node == refNode){
25174             return false;
25175         }
25176
25177         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25178             return false;
25179         }
25180         var index = this.childNodes.indexOf(refNode);
25181         var oldParent = node.parentNode;
25182         var refIndex = index;
25183
25184         // when moving internally, indexes will change after remove
25185         if(oldParent == this && this.childNodes.indexOf(node) < index){
25186             refIndex--;
25187         }
25188
25189         // it's a move, make sure we move it cleanly
25190         if(oldParent){
25191             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25192                 return false;
25193             }
25194             oldParent.removeChild(node);
25195         }
25196         if(refIndex == 0){
25197             this.setFirstChild(node);
25198         }
25199         this.childNodes.splice(refIndex, 0, node);
25200         node.parentNode = this;
25201         var ps = this.childNodes[refIndex-1];
25202         if(ps){
25203             node.previousSibling = ps;
25204             ps.nextSibling = node;
25205         }else{
25206             node.previousSibling = null;
25207         }
25208         node.nextSibling = refNode;
25209         refNode.previousSibling = node;
25210         node.setOwnerTree(this.getOwnerTree());
25211         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25212         if(oldParent){
25213             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25214         }
25215         return node;
25216     },
25217
25218     /**
25219      * Returns the child node at the specified index.
25220      * @param {Number} index
25221      * @return {Node}
25222      */
25223     item : function(index){
25224         return this.childNodes[index];
25225     },
25226
25227     /**
25228      * Replaces one child node in this node with another.
25229      * @param {Node} newChild The replacement node
25230      * @param {Node} oldChild The node to replace
25231      * @return {Node} The replaced node
25232      */
25233     replaceChild : function(newChild, oldChild){
25234         this.insertBefore(newChild, oldChild);
25235         this.removeChild(oldChild);
25236         return oldChild;
25237     },
25238
25239     /**
25240      * Returns the index of a child node
25241      * @param {Node} node
25242      * @return {Number} The index of the node or -1 if it was not found
25243      */
25244     indexOf : function(child){
25245         return this.childNodes.indexOf(child);
25246     },
25247
25248     /**
25249      * Returns the tree this node is in.
25250      * @return {Tree}
25251      */
25252     getOwnerTree : function(){
25253         // if it doesn't have one, look for one
25254         if(!this.ownerTree){
25255             var p = this;
25256             while(p){
25257                 if(p.ownerTree){
25258                     this.ownerTree = p.ownerTree;
25259                     break;
25260                 }
25261                 p = p.parentNode;
25262             }
25263         }
25264         return this.ownerTree;
25265     },
25266
25267     /**
25268      * Returns depth of this node (the root node has a depth of 0)
25269      * @return {Number}
25270      */
25271     getDepth : function(){
25272         var depth = 0;
25273         var p = this;
25274         while(p.parentNode){
25275             ++depth;
25276             p = p.parentNode;
25277         }
25278         return depth;
25279     },
25280
25281     // private
25282     setOwnerTree : function(tree){
25283         // if it's move, we need to update everyone
25284         if(tree != this.ownerTree){
25285             if(this.ownerTree){
25286                 this.ownerTree.unregisterNode(this);
25287             }
25288             this.ownerTree = tree;
25289             var cs = this.childNodes;
25290             for(var i = 0, len = cs.length; i < len; i++) {
25291                 cs[i].setOwnerTree(tree);
25292             }
25293             if(tree){
25294                 tree.registerNode(this);
25295             }
25296         }
25297     },
25298
25299     /**
25300      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25301      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25302      * @return {String} The path
25303      */
25304     getPath : function(attr){
25305         attr = attr || "id";
25306         var p = this.parentNode;
25307         var b = [this.attributes[attr]];
25308         while(p){
25309             b.unshift(p.attributes[attr]);
25310             p = p.parentNode;
25311         }
25312         var sep = this.getOwnerTree().pathSeparator;
25313         return sep + b.join(sep);
25314     },
25315
25316     /**
25317      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25318      * function call will be the scope provided or the current node. The arguments to the function
25319      * will be the args provided or the current node. If the function returns false at any point,
25320      * the bubble is stopped.
25321      * @param {Function} fn The function to call
25322      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25323      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25324      */
25325     bubble : function(fn, scope, args){
25326         var p = this;
25327         while(p){
25328             if(fn.call(scope || p, args || p) === false){
25329                 break;
25330             }
25331             p = p.parentNode;
25332         }
25333     },
25334
25335     /**
25336      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25337      * function call will be the scope provided or the current node. The arguments to the function
25338      * will be the args provided or the current node. If the function returns false at any point,
25339      * the cascade is stopped on that branch.
25340      * @param {Function} fn The function to call
25341      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25342      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25343      */
25344     cascade : function(fn, scope, args){
25345         if(fn.call(scope || this, args || this) !== false){
25346             var cs = this.childNodes;
25347             for(var i = 0, len = cs.length; i < len; i++) {
25348                 cs[i].cascade(fn, scope, args);
25349             }
25350         }
25351     },
25352
25353     /**
25354      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25355      * function call will be the scope provided or the current node. The arguments to the function
25356      * will be the args provided or the current node. If the function returns false at any point,
25357      * the iteration stops.
25358      * @param {Function} fn The function to call
25359      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25360      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25361      */
25362     eachChild : function(fn, scope, args){
25363         var cs = this.childNodes;
25364         for(var i = 0, len = cs.length; i < len; i++) {
25365                 if(fn.call(scope || this, args || cs[i]) === false){
25366                     break;
25367                 }
25368         }
25369     },
25370
25371     /**
25372      * Finds the first child that has the attribute with the specified value.
25373      * @param {String} attribute The attribute name
25374      * @param {Mixed} value The value to search for
25375      * @return {Node} The found child or null if none was found
25376      */
25377     findChild : function(attribute, value){
25378         var cs = this.childNodes;
25379         for(var i = 0, len = cs.length; i < len; i++) {
25380                 if(cs[i].attributes[attribute] == value){
25381                     return cs[i];
25382                 }
25383         }
25384         return null;
25385     },
25386
25387     /**
25388      * Finds the first child by a custom function. The child matches if the function passed
25389      * returns true.
25390      * @param {Function} fn
25391      * @param {Object} scope (optional)
25392      * @return {Node} The found child or null if none was found
25393      */
25394     findChildBy : function(fn, scope){
25395         var cs = this.childNodes;
25396         for(var i = 0, len = cs.length; i < len; i++) {
25397                 if(fn.call(scope||cs[i], cs[i]) === true){
25398                     return cs[i];
25399                 }
25400         }
25401         return null;
25402     },
25403
25404     /**
25405      * Sorts this nodes children using the supplied sort function
25406      * @param {Function} fn
25407      * @param {Object} scope (optional)
25408      */
25409     sort : function(fn, scope){
25410         var cs = this.childNodes;
25411         var len = cs.length;
25412         if(len > 0){
25413             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25414             cs.sort(sortFn);
25415             for(var i = 0; i < len; i++){
25416                 var n = cs[i];
25417                 n.previousSibling = cs[i-1];
25418                 n.nextSibling = cs[i+1];
25419                 if(i == 0){
25420                     this.setFirstChild(n);
25421                 }
25422                 if(i == len-1){
25423                     this.setLastChild(n);
25424                 }
25425             }
25426         }
25427     },
25428
25429     /**
25430      * Returns true if this node is an ancestor (at any point) of the passed node.
25431      * @param {Node} node
25432      * @return {Boolean}
25433      */
25434     contains : function(node){
25435         return node.isAncestor(this);
25436     },
25437
25438     /**
25439      * Returns true if the passed node is an ancestor (at any point) of this node.
25440      * @param {Node} node
25441      * @return {Boolean}
25442      */
25443     isAncestor : function(node){
25444         var p = this.parentNode;
25445         while(p){
25446             if(p == node){
25447                 return true;
25448             }
25449             p = p.parentNode;
25450         }
25451         return false;
25452     },
25453
25454     toString : function(){
25455         return "[Node"+(this.id?" "+this.id:"")+"]";
25456     }
25457 });/*
25458  * Based on:
25459  * Ext JS Library 1.1.1
25460  * Copyright(c) 2006-2007, Ext JS, LLC.
25461  *
25462  * Originally Released Under LGPL - original licence link has changed is not relivant.
25463  *
25464  * Fork - LGPL
25465  * <script type="text/javascript">
25466  */
25467  (function(){ 
25468 /**
25469  * @class Roo.Layer
25470  * @extends Roo.Element
25471  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25472  * automatic maintaining of shadow/shim positions.
25473  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25474  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25475  * you can pass a string with a CSS class name. False turns off the shadow.
25476  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25477  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25478  * @cfg {String} cls CSS class to add to the element
25479  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25480  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25481  * @constructor
25482  * @param {Object} config An object with config options.
25483  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25484  */
25485
25486 Roo.Layer = function(config, existingEl){
25487     config = config || {};
25488     var dh = Roo.DomHelper;
25489     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25490     if(existingEl){
25491         this.dom = Roo.getDom(existingEl);
25492     }
25493     if(!this.dom){
25494         var o = config.dh || {tag: "div", cls: "x-layer"};
25495         this.dom = dh.append(pel, o);
25496     }
25497     if(config.cls){
25498         this.addClass(config.cls);
25499     }
25500     this.constrain = config.constrain !== false;
25501     this.visibilityMode = Roo.Element.VISIBILITY;
25502     if(config.id){
25503         this.id = this.dom.id = config.id;
25504     }else{
25505         this.id = Roo.id(this.dom);
25506     }
25507     this.zindex = config.zindex || this.getZIndex();
25508     this.position("absolute", this.zindex);
25509     if(config.shadow){
25510         this.shadowOffset = config.shadowOffset || 4;
25511         this.shadow = new Roo.Shadow({
25512             offset : this.shadowOffset,
25513             mode : config.shadow
25514         });
25515     }else{
25516         this.shadowOffset = 0;
25517     }
25518     this.useShim = config.shim !== false && Roo.useShims;
25519     this.useDisplay = config.useDisplay;
25520     this.hide();
25521 };
25522
25523 var supr = Roo.Element.prototype;
25524
25525 // shims are shared among layer to keep from having 100 iframes
25526 var shims = [];
25527
25528 Roo.extend(Roo.Layer, Roo.Element, {
25529
25530     getZIndex : function(){
25531         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25532     },
25533
25534     getShim : function(){
25535         if(!this.useShim){
25536             return null;
25537         }
25538         if(this.shim){
25539             return this.shim;
25540         }
25541         var shim = shims.shift();
25542         if(!shim){
25543             shim = this.createShim();
25544             shim.enableDisplayMode('block');
25545             shim.dom.style.display = 'none';
25546             shim.dom.style.visibility = 'visible';
25547         }
25548         var pn = this.dom.parentNode;
25549         if(shim.dom.parentNode != pn){
25550             pn.insertBefore(shim.dom, this.dom);
25551         }
25552         shim.setStyle('z-index', this.getZIndex()-2);
25553         this.shim = shim;
25554         return shim;
25555     },
25556
25557     hideShim : function(){
25558         if(this.shim){
25559             this.shim.setDisplayed(false);
25560             shims.push(this.shim);
25561             delete this.shim;
25562         }
25563     },
25564
25565     disableShadow : function(){
25566         if(this.shadow){
25567             this.shadowDisabled = true;
25568             this.shadow.hide();
25569             this.lastShadowOffset = this.shadowOffset;
25570             this.shadowOffset = 0;
25571         }
25572     },
25573
25574     enableShadow : function(show){
25575         if(this.shadow){
25576             this.shadowDisabled = false;
25577             this.shadowOffset = this.lastShadowOffset;
25578             delete this.lastShadowOffset;
25579             if(show){
25580                 this.sync(true);
25581             }
25582         }
25583     },
25584
25585     // private
25586     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25587     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25588     sync : function(doShow){
25589         var sw = this.shadow;
25590         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25591             var sh = this.getShim();
25592
25593             var w = this.getWidth(),
25594                 h = this.getHeight();
25595
25596             var l = this.getLeft(true),
25597                 t = this.getTop(true);
25598
25599             if(sw && !this.shadowDisabled){
25600                 if(doShow && !sw.isVisible()){
25601                     sw.show(this);
25602                 }else{
25603                     sw.realign(l, t, w, h);
25604                 }
25605                 if(sh){
25606                     if(doShow){
25607                        sh.show();
25608                     }
25609                     // fit the shim behind the shadow, so it is shimmed too
25610                     var a = sw.adjusts, s = sh.dom.style;
25611                     s.left = (Math.min(l, l+a.l))+"px";
25612                     s.top = (Math.min(t, t+a.t))+"px";
25613                     s.width = (w+a.w)+"px";
25614                     s.height = (h+a.h)+"px";
25615                 }
25616             }else if(sh){
25617                 if(doShow){
25618                    sh.show();
25619                 }
25620                 sh.setSize(w, h);
25621                 sh.setLeftTop(l, t);
25622             }
25623             
25624         }
25625     },
25626
25627     // private
25628     destroy : function(){
25629         this.hideShim();
25630         if(this.shadow){
25631             this.shadow.hide();
25632         }
25633         this.removeAllListeners();
25634         var pn = this.dom.parentNode;
25635         if(pn){
25636             pn.removeChild(this.dom);
25637         }
25638         Roo.Element.uncache(this.id);
25639     },
25640
25641     remove : function(){
25642         this.destroy();
25643     },
25644
25645     // private
25646     beginUpdate : function(){
25647         this.updating = true;
25648     },
25649
25650     // private
25651     endUpdate : function(){
25652         this.updating = false;
25653         this.sync(true);
25654     },
25655
25656     // private
25657     hideUnders : function(negOffset){
25658         if(this.shadow){
25659             this.shadow.hide();
25660         }
25661         this.hideShim();
25662     },
25663
25664     // private
25665     constrainXY : function(){
25666         if(this.constrain){
25667             var vw = Roo.lib.Dom.getViewWidth(),
25668                 vh = Roo.lib.Dom.getViewHeight();
25669             var s = Roo.get(document).getScroll();
25670
25671             var xy = this.getXY();
25672             var x = xy[0], y = xy[1];   
25673             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25674             // only move it if it needs it
25675             var moved = false;
25676             // first validate right/bottom
25677             if((x + w) > vw+s.left){
25678                 x = vw - w - this.shadowOffset;
25679                 moved = true;
25680             }
25681             if((y + h) > vh+s.top){
25682                 y = vh - h - this.shadowOffset;
25683                 moved = true;
25684             }
25685             // then make sure top/left isn't negative
25686             if(x < s.left){
25687                 x = s.left;
25688                 moved = true;
25689             }
25690             if(y < s.top){
25691                 y = s.top;
25692                 moved = true;
25693             }
25694             if(moved){
25695                 if(this.avoidY){
25696                     var ay = this.avoidY;
25697                     if(y <= ay && (y+h) >= ay){
25698                         y = ay-h-5;   
25699                     }
25700                 }
25701                 xy = [x, y];
25702                 this.storeXY(xy);
25703                 supr.setXY.call(this, xy);
25704                 this.sync();
25705             }
25706         }
25707     },
25708
25709     isVisible : function(){
25710         return this.visible;    
25711     },
25712
25713     // private
25714     showAction : function(){
25715         this.visible = true; // track visibility to prevent getStyle calls
25716         if(this.useDisplay === true){
25717             this.setDisplayed("");
25718         }else if(this.lastXY){
25719             supr.setXY.call(this, this.lastXY);
25720         }else if(this.lastLT){
25721             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25722         }
25723     },
25724
25725     // private
25726     hideAction : function(){
25727         this.visible = false;
25728         if(this.useDisplay === true){
25729             this.setDisplayed(false);
25730         }else{
25731             this.setLeftTop(-10000,-10000);
25732         }
25733     },
25734
25735     // overridden Element method
25736     setVisible : function(v, a, d, c, e){
25737         if(v){
25738             this.showAction();
25739         }
25740         if(a && v){
25741             var cb = function(){
25742                 this.sync(true);
25743                 if(c){
25744                     c();
25745                 }
25746             }.createDelegate(this);
25747             supr.setVisible.call(this, true, true, d, cb, e);
25748         }else{
25749             if(!v){
25750                 this.hideUnders(true);
25751             }
25752             var cb = c;
25753             if(a){
25754                 cb = function(){
25755                     this.hideAction();
25756                     if(c){
25757                         c();
25758                     }
25759                 }.createDelegate(this);
25760             }
25761             supr.setVisible.call(this, v, a, d, cb, e);
25762             if(v){
25763                 this.sync(true);
25764             }else if(!a){
25765                 this.hideAction();
25766             }
25767         }
25768     },
25769
25770     storeXY : function(xy){
25771         delete this.lastLT;
25772         this.lastXY = xy;
25773     },
25774
25775     storeLeftTop : function(left, top){
25776         delete this.lastXY;
25777         this.lastLT = [left, top];
25778     },
25779
25780     // private
25781     beforeFx : function(){
25782         this.beforeAction();
25783         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25784     },
25785
25786     // private
25787     afterFx : function(){
25788         Roo.Layer.superclass.afterFx.apply(this, arguments);
25789         this.sync(this.isVisible());
25790     },
25791
25792     // private
25793     beforeAction : function(){
25794         if(!this.updating && this.shadow){
25795             this.shadow.hide();
25796         }
25797     },
25798
25799     // overridden Element method
25800     setLeft : function(left){
25801         this.storeLeftTop(left, this.getTop(true));
25802         supr.setLeft.apply(this, arguments);
25803         this.sync();
25804     },
25805
25806     setTop : function(top){
25807         this.storeLeftTop(this.getLeft(true), top);
25808         supr.setTop.apply(this, arguments);
25809         this.sync();
25810     },
25811
25812     setLeftTop : function(left, top){
25813         this.storeLeftTop(left, top);
25814         supr.setLeftTop.apply(this, arguments);
25815         this.sync();
25816     },
25817
25818     setXY : function(xy, a, d, c, e){
25819         this.fixDisplay();
25820         this.beforeAction();
25821         this.storeXY(xy);
25822         var cb = this.createCB(c);
25823         supr.setXY.call(this, xy, a, d, cb, e);
25824         if(!a){
25825             cb();
25826         }
25827     },
25828
25829     // private
25830     createCB : function(c){
25831         var el = this;
25832         return function(){
25833             el.constrainXY();
25834             el.sync(true);
25835             if(c){
25836                 c();
25837             }
25838         };
25839     },
25840
25841     // overridden Element method
25842     setX : function(x, a, d, c, e){
25843         this.setXY([x, this.getY()], a, d, c, e);
25844     },
25845
25846     // overridden Element method
25847     setY : function(y, a, d, c, e){
25848         this.setXY([this.getX(), y], a, d, c, e);
25849     },
25850
25851     // overridden Element method
25852     setSize : function(w, h, a, d, c, e){
25853         this.beforeAction();
25854         var cb = this.createCB(c);
25855         supr.setSize.call(this, w, h, a, d, cb, e);
25856         if(!a){
25857             cb();
25858         }
25859     },
25860
25861     // overridden Element method
25862     setWidth : function(w, a, d, c, e){
25863         this.beforeAction();
25864         var cb = this.createCB(c);
25865         supr.setWidth.call(this, w, a, d, cb, e);
25866         if(!a){
25867             cb();
25868         }
25869     },
25870
25871     // overridden Element method
25872     setHeight : function(h, a, d, c, e){
25873         this.beforeAction();
25874         var cb = this.createCB(c);
25875         supr.setHeight.call(this, h, a, d, cb, e);
25876         if(!a){
25877             cb();
25878         }
25879     },
25880
25881     // overridden Element method
25882     setBounds : function(x, y, w, h, a, d, c, e){
25883         this.beforeAction();
25884         var cb = this.createCB(c);
25885         if(!a){
25886             this.storeXY([x, y]);
25887             supr.setXY.call(this, [x, y]);
25888             supr.setSize.call(this, w, h, a, d, cb, e);
25889             cb();
25890         }else{
25891             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25892         }
25893         return this;
25894     },
25895     
25896     /**
25897      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25898      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25899      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25900      * @param {Number} zindex The new z-index to set
25901      * @return {this} The Layer
25902      */
25903     setZIndex : function(zindex){
25904         this.zindex = zindex;
25905         this.setStyle("z-index", zindex + 2);
25906         if(this.shadow){
25907             this.shadow.setZIndex(zindex + 1);
25908         }
25909         if(this.shim){
25910             this.shim.setStyle("z-index", zindex);
25911         }
25912     }
25913 });
25914 })();/*
25915  * Based on:
25916  * Ext JS Library 1.1.1
25917  * Copyright(c) 2006-2007, Ext JS, LLC.
25918  *
25919  * Originally Released Under LGPL - original licence link has changed is not relivant.
25920  *
25921  * Fork - LGPL
25922  * <script type="text/javascript">
25923  */
25924
25925
25926 /**
25927  * @class Roo.Shadow
25928  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25929  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25930  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25931  * @constructor
25932  * Create a new Shadow
25933  * @param {Object} config The config object
25934  */
25935 Roo.Shadow = function(config){
25936     Roo.apply(this, config);
25937     if(typeof this.mode != "string"){
25938         this.mode = this.defaultMode;
25939     }
25940     var o = this.offset, a = {h: 0};
25941     var rad = Math.floor(this.offset/2);
25942     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25943         case "drop":
25944             a.w = 0;
25945             a.l = a.t = o;
25946             a.t -= 1;
25947             if(Roo.isIE){
25948                 a.l -= this.offset + rad;
25949                 a.t -= this.offset + rad;
25950                 a.w -= rad;
25951                 a.h -= rad;
25952                 a.t += 1;
25953             }
25954         break;
25955         case "sides":
25956             a.w = (o*2);
25957             a.l = -o;
25958             a.t = o-1;
25959             if(Roo.isIE){
25960                 a.l -= (this.offset - rad);
25961                 a.t -= this.offset + rad;
25962                 a.l += 1;
25963                 a.w -= (this.offset - rad)*2;
25964                 a.w -= rad + 1;
25965                 a.h -= 1;
25966             }
25967         break;
25968         case "frame":
25969             a.w = a.h = (o*2);
25970             a.l = a.t = -o;
25971             a.t += 1;
25972             a.h -= 2;
25973             if(Roo.isIE){
25974                 a.l -= (this.offset - rad);
25975                 a.t -= (this.offset - rad);
25976                 a.l += 1;
25977                 a.w -= (this.offset + rad + 1);
25978                 a.h -= (this.offset + rad);
25979                 a.h += 1;
25980             }
25981         break;
25982     };
25983
25984     this.adjusts = a;
25985 };
25986
25987 Roo.Shadow.prototype = {
25988     /**
25989      * @cfg {String} mode
25990      * The shadow display mode.  Supports the following options:<br />
25991      * sides: Shadow displays on both sides and bottom only<br />
25992      * frame: Shadow displays equally on all four sides<br />
25993      * drop: Traditional bottom-right drop shadow (default)
25994      */
25995     /**
25996      * @cfg {String} offset
25997      * The number of pixels to offset the shadow from the element (defaults to 4)
25998      */
25999     offset: 4,
26000
26001     // private
26002     defaultMode: "drop",
26003
26004     /**
26005      * Displays the shadow under the target element
26006      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
26007      */
26008     show : function(target){
26009         target = Roo.get(target);
26010         if(!this.el){
26011             this.el = Roo.Shadow.Pool.pull();
26012             if(this.el.dom.nextSibling != target.dom){
26013                 this.el.insertBefore(target);
26014             }
26015         }
26016         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
26017         if(Roo.isIE){
26018             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
26019         }
26020         this.realign(
26021             target.getLeft(true),
26022             target.getTop(true),
26023             target.getWidth(),
26024             target.getHeight()
26025         );
26026         this.el.dom.style.display = "block";
26027     },
26028
26029     /**
26030      * Returns true if the shadow is visible, else false
26031      */
26032     isVisible : function(){
26033         return this.el ? true : false;  
26034     },
26035
26036     /**
26037      * Direct alignment when values are already available. Show must be called at least once before
26038      * calling this method to ensure it is initialized.
26039      * @param {Number} left The target element left position
26040      * @param {Number} top The target element top position
26041      * @param {Number} width The target element width
26042      * @param {Number} height The target element height
26043      */
26044     realign : function(l, t, w, h){
26045         if(!this.el){
26046             return;
26047         }
26048         var a = this.adjusts, d = this.el.dom, s = d.style;
26049         var iea = 0;
26050         s.left = (l+a.l)+"px";
26051         s.top = (t+a.t)+"px";
26052         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26053  
26054         if(s.width != sws || s.height != shs){
26055             s.width = sws;
26056             s.height = shs;
26057             if(!Roo.isIE){
26058                 var cn = d.childNodes;
26059                 var sww = Math.max(0, (sw-12))+"px";
26060                 cn[0].childNodes[1].style.width = sww;
26061                 cn[1].childNodes[1].style.width = sww;
26062                 cn[2].childNodes[1].style.width = sww;
26063                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26064             }
26065         }
26066     },
26067
26068     /**
26069      * Hides this shadow
26070      */
26071     hide : function(){
26072         if(this.el){
26073             this.el.dom.style.display = "none";
26074             Roo.Shadow.Pool.push(this.el);
26075             delete this.el;
26076         }
26077     },
26078
26079     /**
26080      * Adjust the z-index of this shadow
26081      * @param {Number} zindex The new z-index
26082      */
26083     setZIndex : function(z){
26084         this.zIndex = z;
26085         if(this.el){
26086             this.el.setStyle("z-index", z);
26087         }
26088     }
26089 };
26090
26091 // Private utility class that manages the internal Shadow cache
26092 Roo.Shadow.Pool = function(){
26093     var p = [];
26094     var markup = Roo.isIE ?
26095                  '<div class="x-ie-shadow"></div>' :
26096                  '<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>';
26097     return {
26098         pull : function(){
26099             var sh = p.shift();
26100             if(!sh){
26101                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26102                 sh.autoBoxAdjust = false;
26103             }
26104             return sh;
26105         },
26106
26107         push : function(sh){
26108             p.push(sh);
26109         }
26110     };
26111 }();/*
26112  * Based on:
26113  * Ext JS Library 1.1.1
26114  * Copyright(c) 2006-2007, Ext JS, LLC.
26115  *
26116  * Originally Released Under LGPL - original licence link has changed is not relivant.
26117  *
26118  * Fork - LGPL
26119  * <script type="text/javascript">
26120  */
26121
26122
26123 /**
26124  * @class Roo.SplitBar
26125  * @extends Roo.util.Observable
26126  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26127  * <br><br>
26128  * Usage:
26129  * <pre><code>
26130 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26131                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26132 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26133 split.minSize = 100;
26134 split.maxSize = 600;
26135 split.animate = true;
26136 split.on('moved', splitterMoved);
26137 </code></pre>
26138  * @constructor
26139  * Create a new SplitBar
26140  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26141  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26142  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26143  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26144                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26145                         position of the SplitBar).
26146  */
26147 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26148     
26149     /** @private */
26150     this.el = Roo.get(dragElement, true);
26151     this.el.dom.unselectable = "on";
26152     /** @private */
26153     this.resizingEl = Roo.get(resizingElement, true);
26154
26155     /**
26156      * @private
26157      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26158      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26159      * @type Number
26160      */
26161     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26162     
26163     /**
26164      * The minimum size of the resizing element. (Defaults to 0)
26165      * @type Number
26166      */
26167     this.minSize = 0;
26168     
26169     /**
26170      * The maximum size of the resizing element. (Defaults to 2000)
26171      * @type Number
26172      */
26173     this.maxSize = 2000;
26174     
26175     /**
26176      * Whether to animate the transition to the new size
26177      * @type Boolean
26178      */
26179     this.animate = false;
26180     
26181     /**
26182      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26183      * @type Boolean
26184      */
26185     this.useShim = false;
26186     
26187     /** @private */
26188     this.shim = null;
26189     
26190     if(!existingProxy){
26191         /** @private */
26192         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26193     }else{
26194         this.proxy = Roo.get(existingProxy).dom;
26195     }
26196     /** @private */
26197     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26198     
26199     /** @private */
26200     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26201     
26202     /** @private */
26203     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26204     
26205     /** @private */
26206     this.dragSpecs = {};
26207     
26208     /**
26209      * @private The adapter to use to positon and resize elements
26210      */
26211     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26212     this.adapter.init(this);
26213     
26214     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26215         /** @private */
26216         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26217         this.el.addClass("x-splitbar-h");
26218     }else{
26219         /** @private */
26220         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26221         this.el.addClass("x-splitbar-v");
26222     }
26223     
26224     this.addEvents({
26225         /**
26226          * @event resize
26227          * Fires when the splitter is moved (alias for {@link #event-moved})
26228          * @param {Roo.SplitBar} this
26229          * @param {Number} newSize the new width or height
26230          */
26231         "resize" : true,
26232         /**
26233          * @event moved
26234          * Fires when the splitter is moved
26235          * @param {Roo.SplitBar} this
26236          * @param {Number} newSize the new width or height
26237          */
26238         "moved" : true,
26239         /**
26240          * @event beforeresize
26241          * Fires before the splitter is dragged
26242          * @param {Roo.SplitBar} this
26243          */
26244         "beforeresize" : true,
26245
26246         "beforeapply" : true
26247     });
26248
26249     Roo.util.Observable.call(this);
26250 };
26251
26252 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26253     onStartProxyDrag : function(x, y){
26254         this.fireEvent("beforeresize", this);
26255         if(!this.overlay){
26256             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26257             o.unselectable();
26258             o.enableDisplayMode("block");
26259             // all splitbars share the same overlay
26260             Roo.SplitBar.prototype.overlay = o;
26261         }
26262         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26263         this.overlay.show();
26264         Roo.get(this.proxy).setDisplayed("block");
26265         var size = this.adapter.getElementSize(this);
26266         this.activeMinSize = this.getMinimumSize();;
26267         this.activeMaxSize = this.getMaximumSize();;
26268         var c1 = size - this.activeMinSize;
26269         var c2 = Math.max(this.activeMaxSize - size, 0);
26270         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26271             this.dd.resetConstraints();
26272             this.dd.setXConstraint(
26273                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26274                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26275             );
26276             this.dd.setYConstraint(0, 0);
26277         }else{
26278             this.dd.resetConstraints();
26279             this.dd.setXConstraint(0, 0);
26280             this.dd.setYConstraint(
26281                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26282                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26283             );
26284          }
26285         this.dragSpecs.startSize = size;
26286         this.dragSpecs.startPoint = [x, y];
26287         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26288     },
26289     
26290     /** 
26291      * @private Called after the drag operation by the DDProxy
26292      */
26293     onEndProxyDrag : function(e){
26294         Roo.get(this.proxy).setDisplayed(false);
26295         var endPoint = Roo.lib.Event.getXY(e);
26296         if(this.overlay){
26297             this.overlay.hide();
26298         }
26299         var newSize;
26300         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26301             newSize = this.dragSpecs.startSize + 
26302                 (this.placement == Roo.SplitBar.LEFT ?
26303                     endPoint[0] - this.dragSpecs.startPoint[0] :
26304                     this.dragSpecs.startPoint[0] - endPoint[0]
26305                 );
26306         }else{
26307             newSize = this.dragSpecs.startSize + 
26308                 (this.placement == Roo.SplitBar.TOP ?
26309                     endPoint[1] - this.dragSpecs.startPoint[1] :
26310                     this.dragSpecs.startPoint[1] - endPoint[1]
26311                 );
26312         }
26313         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26314         if(newSize != this.dragSpecs.startSize){
26315             if(this.fireEvent('beforeapply', this, newSize) !== false){
26316                 this.adapter.setElementSize(this, newSize);
26317                 this.fireEvent("moved", this, newSize);
26318                 this.fireEvent("resize", this, newSize);
26319             }
26320         }
26321     },
26322     
26323     /**
26324      * Get the adapter this SplitBar uses
26325      * @return The adapter object
26326      */
26327     getAdapter : function(){
26328         return this.adapter;
26329     },
26330     
26331     /**
26332      * Set the adapter this SplitBar uses
26333      * @param {Object} adapter A SplitBar adapter object
26334      */
26335     setAdapter : function(adapter){
26336         this.adapter = adapter;
26337         this.adapter.init(this);
26338     },
26339     
26340     /**
26341      * Gets the minimum size for the resizing element
26342      * @return {Number} The minimum size
26343      */
26344     getMinimumSize : function(){
26345         return this.minSize;
26346     },
26347     
26348     /**
26349      * Sets the minimum size for the resizing element
26350      * @param {Number} minSize The minimum size
26351      */
26352     setMinimumSize : function(minSize){
26353         this.minSize = minSize;
26354     },
26355     
26356     /**
26357      * Gets the maximum size for the resizing element
26358      * @return {Number} The maximum size
26359      */
26360     getMaximumSize : function(){
26361         return this.maxSize;
26362     },
26363     
26364     /**
26365      * Sets the maximum size for the resizing element
26366      * @param {Number} maxSize The maximum size
26367      */
26368     setMaximumSize : function(maxSize){
26369         this.maxSize = maxSize;
26370     },
26371     
26372     /**
26373      * Sets the initialize size for the resizing element
26374      * @param {Number} size The initial size
26375      */
26376     setCurrentSize : function(size){
26377         var oldAnimate = this.animate;
26378         this.animate = false;
26379         this.adapter.setElementSize(this, size);
26380         this.animate = oldAnimate;
26381     },
26382     
26383     /**
26384      * Destroy this splitbar. 
26385      * @param {Boolean} removeEl True to remove the element
26386      */
26387     destroy : function(removeEl){
26388         if(this.shim){
26389             this.shim.remove();
26390         }
26391         this.dd.unreg();
26392         this.proxy.parentNode.removeChild(this.proxy);
26393         if(removeEl){
26394             this.el.remove();
26395         }
26396     }
26397 });
26398
26399 /**
26400  * @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.
26401  */
26402 Roo.SplitBar.createProxy = function(dir){
26403     var proxy = new Roo.Element(document.createElement("div"));
26404     proxy.unselectable();
26405     var cls = 'x-splitbar-proxy';
26406     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26407     document.body.appendChild(proxy.dom);
26408     return proxy.dom;
26409 };
26410
26411 /** 
26412  * @class Roo.SplitBar.BasicLayoutAdapter
26413  * Default Adapter. It assumes the splitter and resizing element are not positioned
26414  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26415  */
26416 Roo.SplitBar.BasicLayoutAdapter = function(){
26417 };
26418
26419 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26420     // do nothing for now
26421     init : function(s){
26422     
26423     },
26424     /**
26425      * Called before drag operations to get the current size of the resizing element. 
26426      * @param {Roo.SplitBar} s The SplitBar using this adapter
26427      */
26428      getElementSize : function(s){
26429         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26430             return s.resizingEl.getWidth();
26431         }else{
26432             return s.resizingEl.getHeight();
26433         }
26434     },
26435     
26436     /**
26437      * Called after drag operations to set the size of the resizing element.
26438      * @param {Roo.SplitBar} s The SplitBar using this adapter
26439      * @param {Number} newSize The new size to set
26440      * @param {Function} onComplete A function to be invoked when resizing is complete
26441      */
26442     setElementSize : function(s, newSize, onComplete){
26443         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26444             if(!s.animate){
26445                 s.resizingEl.setWidth(newSize);
26446                 if(onComplete){
26447                     onComplete(s, newSize);
26448                 }
26449             }else{
26450                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26451             }
26452         }else{
26453             
26454             if(!s.animate){
26455                 s.resizingEl.setHeight(newSize);
26456                 if(onComplete){
26457                     onComplete(s, newSize);
26458                 }
26459             }else{
26460                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26461             }
26462         }
26463     }
26464 };
26465
26466 /** 
26467  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26468  * @extends Roo.SplitBar.BasicLayoutAdapter
26469  * Adapter that  moves the splitter element to align with the resized sizing element. 
26470  * Used with an absolute positioned SplitBar.
26471  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26472  * document.body, make sure you assign an id to the body element.
26473  */
26474 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26475     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26476     this.container = Roo.get(container);
26477 };
26478
26479 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26480     init : function(s){
26481         this.basic.init(s);
26482     },
26483     
26484     getElementSize : function(s){
26485         return this.basic.getElementSize(s);
26486     },
26487     
26488     setElementSize : function(s, newSize, onComplete){
26489         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26490     },
26491     
26492     moveSplitter : function(s){
26493         var yes = Roo.SplitBar;
26494         switch(s.placement){
26495             case yes.LEFT:
26496                 s.el.setX(s.resizingEl.getRight());
26497                 break;
26498             case yes.RIGHT:
26499                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26500                 break;
26501             case yes.TOP:
26502                 s.el.setY(s.resizingEl.getBottom());
26503                 break;
26504             case yes.BOTTOM:
26505                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26506                 break;
26507         }
26508     }
26509 };
26510
26511 /**
26512  * Orientation constant - Create a vertical SplitBar
26513  * @static
26514  * @type Number
26515  */
26516 Roo.SplitBar.VERTICAL = 1;
26517
26518 /**
26519  * Orientation constant - Create a horizontal SplitBar
26520  * @static
26521  * @type Number
26522  */
26523 Roo.SplitBar.HORIZONTAL = 2;
26524
26525 /**
26526  * Placement constant - The resizing element is to the left of the splitter element
26527  * @static
26528  * @type Number
26529  */
26530 Roo.SplitBar.LEFT = 1;
26531
26532 /**
26533  * Placement constant - The resizing element is to the right of the splitter element
26534  * @static
26535  * @type Number
26536  */
26537 Roo.SplitBar.RIGHT = 2;
26538
26539 /**
26540  * Placement constant - The resizing element is positioned above the splitter element
26541  * @static
26542  * @type Number
26543  */
26544 Roo.SplitBar.TOP = 3;
26545
26546 /**
26547  * Placement constant - The resizing element is positioned under splitter element
26548  * @static
26549  * @type Number
26550  */
26551 Roo.SplitBar.BOTTOM = 4;
26552 /*
26553  * Based on:
26554  * Ext JS Library 1.1.1
26555  * Copyright(c) 2006-2007, Ext JS, LLC.
26556  *
26557  * Originally Released Under LGPL - original licence link has changed is not relivant.
26558  *
26559  * Fork - LGPL
26560  * <script type="text/javascript">
26561  */
26562
26563 /**
26564  * @class Roo.View
26565  * @extends Roo.util.Observable
26566  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26567  * This class also supports single and multi selection modes. <br>
26568  * Create a data model bound view:
26569  <pre><code>
26570  var store = new Roo.data.Store(...);
26571
26572  var view = new Roo.View({
26573     el : "my-element",
26574     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26575  
26576     singleSelect: true,
26577     selectedClass: "ydataview-selected",
26578     store: store
26579  });
26580
26581  // listen for node click?
26582  view.on("click", function(vw, index, node, e){
26583  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26584  });
26585
26586  // load XML data
26587  dataModel.load("foobar.xml");
26588  </code></pre>
26589  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26590  * <br><br>
26591  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26592  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26593  * 
26594  * Note: old style constructor is still suported (container, template, config)
26595  * 
26596  * @constructor
26597  * Create a new View
26598  * @param {Object} config The config object
26599  * 
26600  */
26601 Roo.View = function(config, depreciated_tpl, depreciated_config){
26602     
26603     this.parent = false;
26604     
26605     if (typeof(depreciated_tpl) == 'undefined') {
26606         // new way.. - universal constructor.
26607         Roo.apply(this, config);
26608         this.el  = Roo.get(this.el);
26609     } else {
26610         // old format..
26611         this.el  = Roo.get(config);
26612         this.tpl = depreciated_tpl;
26613         Roo.apply(this, depreciated_config);
26614     }
26615     this.wrapEl  = this.el.wrap().wrap();
26616     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26617     
26618     
26619     if(typeof(this.tpl) == "string"){
26620         this.tpl = new Roo.Template(this.tpl);
26621     } else {
26622         // support xtype ctors..
26623         this.tpl = new Roo.factory(this.tpl, Roo);
26624     }
26625     
26626     
26627     this.tpl.compile();
26628     
26629     /** @private */
26630     this.addEvents({
26631         /**
26632          * @event beforeclick
26633          * Fires before a click is processed. Returns false to cancel the default action.
26634          * @param {Roo.View} this
26635          * @param {Number} index The index of the target node
26636          * @param {HTMLElement} node The target node
26637          * @param {Roo.EventObject} e The raw event object
26638          */
26639             "beforeclick" : true,
26640         /**
26641          * @event click
26642          * Fires when a template node is clicked.
26643          * @param {Roo.View} this
26644          * @param {Number} index The index of the target node
26645          * @param {HTMLElement} node The target node
26646          * @param {Roo.EventObject} e The raw event object
26647          */
26648             "click" : true,
26649         /**
26650          * @event dblclick
26651          * Fires when a template node is double clicked.
26652          * @param {Roo.View} this
26653          * @param {Number} index The index of the target node
26654          * @param {HTMLElement} node The target node
26655          * @param {Roo.EventObject} e The raw event object
26656          */
26657             "dblclick" : true,
26658         /**
26659          * @event contextmenu
26660          * Fires when a template node is right clicked.
26661          * @param {Roo.View} this
26662          * @param {Number} index The index of the target node
26663          * @param {HTMLElement} node The target node
26664          * @param {Roo.EventObject} e The raw event object
26665          */
26666             "contextmenu" : true,
26667         /**
26668          * @event selectionchange
26669          * Fires when the selected nodes change.
26670          * @param {Roo.View} this
26671          * @param {Array} selections Array of the selected nodes
26672          */
26673             "selectionchange" : true,
26674     
26675         /**
26676          * @event beforeselect
26677          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26678          * @param {Roo.View} this
26679          * @param {HTMLElement} node The node to be selected
26680          * @param {Array} selections Array of currently selected nodes
26681          */
26682             "beforeselect" : true,
26683         /**
26684          * @event preparedata
26685          * Fires on every row to render, to allow you to change the data.
26686          * @param {Roo.View} this
26687          * @param {Object} data to be rendered (change this)
26688          */
26689           "preparedata" : true
26690           
26691           
26692         });
26693
26694
26695
26696     this.el.on({
26697         "click": this.onClick,
26698         "dblclick": this.onDblClick,
26699         "contextmenu": this.onContextMenu,
26700         scope:this
26701     });
26702
26703     this.selections = [];
26704     this.nodes = [];
26705     this.cmp = new Roo.CompositeElementLite([]);
26706     if(this.store){
26707         this.store = Roo.factory(this.store, Roo.data);
26708         this.setStore(this.store, true);
26709     }
26710     
26711     if ( this.footer && this.footer.xtype) {
26712            
26713          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26714         
26715         this.footer.dataSource = this.store;
26716         this.footer.container = fctr;
26717         this.footer = Roo.factory(this.footer, Roo);
26718         fctr.insertFirst(this.el);
26719         
26720         // this is a bit insane - as the paging toolbar seems to detach the el..
26721 //        dom.parentNode.parentNode.parentNode
26722          // they get detached?
26723     }
26724     
26725     
26726     Roo.View.superclass.constructor.call(this);
26727     
26728     
26729 };
26730
26731 Roo.extend(Roo.View, Roo.util.Observable, {
26732     
26733      /**
26734      * @cfg {Roo.data.Store} store Data store to load data from.
26735      */
26736     store : false,
26737     
26738     /**
26739      * @cfg {String|Roo.Element} el The container element.
26740      */
26741     el : '',
26742     
26743     /**
26744      * @cfg {String|Roo.Template} tpl The template used by this View 
26745      */
26746     tpl : false,
26747     /**
26748      * @cfg {String} dataName the named area of the template to use as the data area
26749      *                          Works with domtemplates roo-name="name"
26750      */
26751     dataName: false,
26752     /**
26753      * @cfg {String} selectedClass The css class to add to selected nodes
26754      */
26755     selectedClass : "x-view-selected",
26756      /**
26757      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26758      */
26759     emptyText : "",
26760     
26761     /**
26762      * @cfg {String} text to display on mask (default Loading)
26763      */
26764     mask : false,
26765     /**
26766      * @cfg {Boolean} multiSelect Allow multiple selection
26767      */
26768     multiSelect : false,
26769     /**
26770      * @cfg {Boolean} singleSelect Allow single selection
26771      */
26772     singleSelect:  false,
26773     
26774     /**
26775      * @cfg {Boolean} toggleSelect - selecting 
26776      */
26777     toggleSelect : false,
26778     
26779     /**
26780      * @cfg {Boolean} tickable - selecting 
26781      */
26782     tickable : false,
26783     
26784     /**
26785      * Returns the element this view is bound to.
26786      * @return {Roo.Element}
26787      */
26788     getEl : function(){
26789         return this.wrapEl;
26790     },
26791     
26792     
26793
26794     /**
26795      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26796      */
26797     refresh : function(){
26798         //Roo.log('refresh');
26799         var t = this.tpl;
26800         
26801         // if we are using something like 'domtemplate', then
26802         // the what gets used is:
26803         // t.applySubtemplate(NAME, data, wrapping data..)
26804         // the outer template then get' applied with
26805         //     the store 'extra data'
26806         // and the body get's added to the
26807         //      roo-name="data" node?
26808         //      <span class='roo-tpl-{name}'></span> ?????
26809         
26810         
26811         
26812         this.clearSelections();
26813         this.el.update("");
26814         var html = [];
26815         var records = this.store.getRange();
26816         if(records.length < 1) {
26817             
26818             // is this valid??  = should it render a template??
26819             
26820             this.el.update(this.emptyText);
26821             return;
26822         }
26823         var el = this.el;
26824         if (this.dataName) {
26825             this.el.update(t.apply(this.store.meta)); //????
26826             el = this.el.child('.roo-tpl-' + this.dataName);
26827         }
26828         
26829         for(var i = 0, len = records.length; i < len; i++){
26830             var data = this.prepareData(records[i].data, i, records[i]);
26831             this.fireEvent("preparedata", this, data, i, records[i]);
26832             
26833             var d = Roo.apply({}, data);
26834             
26835             if(this.tickable){
26836                 Roo.apply(d, {'roo-id' : Roo.id()});
26837                 
26838                 var _this = this;
26839             
26840                 Roo.each(this.parent.item, function(item){
26841                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26842                         return;
26843                     }
26844                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26845                 });
26846             }
26847             
26848             html[html.length] = Roo.util.Format.trim(
26849                 this.dataName ?
26850                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26851                     t.apply(d)
26852             );
26853         }
26854         
26855         
26856         
26857         el.update(html.join(""));
26858         this.nodes = el.dom.childNodes;
26859         this.updateIndexes(0);
26860     },
26861     
26862
26863     /**
26864      * Function to override to reformat the data that is sent to
26865      * the template for each node.
26866      * DEPRICATED - use the preparedata event handler.
26867      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26868      * a JSON object for an UpdateManager bound view).
26869      */
26870     prepareData : function(data, index, record)
26871     {
26872         this.fireEvent("preparedata", this, data, index, record);
26873         return data;
26874     },
26875
26876     onUpdate : function(ds, record){
26877         // Roo.log('on update');   
26878         this.clearSelections();
26879         var index = this.store.indexOf(record);
26880         var n = this.nodes[index];
26881         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26882         n.parentNode.removeChild(n);
26883         this.updateIndexes(index, index);
26884     },
26885
26886     
26887     
26888 // --------- FIXME     
26889     onAdd : function(ds, records, index)
26890     {
26891         //Roo.log(['on Add', ds, records, index] );        
26892         this.clearSelections();
26893         if(this.nodes.length == 0){
26894             this.refresh();
26895             return;
26896         }
26897         var n = this.nodes[index];
26898         for(var i = 0, len = records.length; i < len; i++){
26899             var d = this.prepareData(records[i].data, i, records[i]);
26900             if(n){
26901                 this.tpl.insertBefore(n, d);
26902             }else{
26903                 
26904                 this.tpl.append(this.el, d);
26905             }
26906         }
26907         this.updateIndexes(index);
26908     },
26909
26910     onRemove : function(ds, record, index){
26911        // Roo.log('onRemove');
26912         this.clearSelections();
26913         var el = this.dataName  ?
26914             this.el.child('.roo-tpl-' + this.dataName) :
26915             this.el; 
26916         
26917         el.dom.removeChild(this.nodes[index]);
26918         this.updateIndexes(index);
26919     },
26920
26921     /**
26922      * Refresh an individual node.
26923      * @param {Number} index
26924      */
26925     refreshNode : function(index){
26926         this.onUpdate(this.store, this.store.getAt(index));
26927     },
26928
26929     updateIndexes : function(startIndex, endIndex){
26930         var ns = this.nodes;
26931         startIndex = startIndex || 0;
26932         endIndex = endIndex || ns.length - 1;
26933         for(var i = startIndex; i <= endIndex; i++){
26934             ns[i].nodeIndex = i;
26935         }
26936     },
26937
26938     /**
26939      * Changes the data store this view uses and refresh the view.
26940      * @param {Store} store
26941      */
26942     setStore : function(store, initial){
26943         if(!initial && this.store){
26944             this.store.un("datachanged", this.refresh);
26945             this.store.un("add", this.onAdd);
26946             this.store.un("remove", this.onRemove);
26947             this.store.un("update", this.onUpdate);
26948             this.store.un("clear", this.refresh);
26949             this.store.un("beforeload", this.onBeforeLoad);
26950             this.store.un("load", this.onLoad);
26951             this.store.un("loadexception", this.onLoad);
26952         }
26953         if(store){
26954           
26955             store.on("datachanged", this.refresh, this);
26956             store.on("add", this.onAdd, this);
26957             store.on("remove", this.onRemove, this);
26958             store.on("update", this.onUpdate, this);
26959             store.on("clear", this.refresh, this);
26960             store.on("beforeload", this.onBeforeLoad, this);
26961             store.on("load", this.onLoad, this);
26962             store.on("loadexception", this.onLoad, this);
26963         }
26964         
26965         if(store){
26966             this.refresh();
26967         }
26968     },
26969     /**
26970      * onbeforeLoad - masks the loading area.
26971      *
26972      */
26973     onBeforeLoad : function(store,opts)
26974     {
26975          //Roo.log('onBeforeLoad');   
26976         if (!opts.add) {
26977             this.el.update("");
26978         }
26979         this.el.mask(this.mask ? this.mask : "Loading" ); 
26980     },
26981     onLoad : function ()
26982     {
26983         this.el.unmask();
26984     },
26985     
26986
26987     /**
26988      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26989      * @param {HTMLElement} node
26990      * @return {HTMLElement} The template node
26991      */
26992     findItemFromChild : function(node){
26993         var el = this.dataName  ?
26994             this.el.child('.roo-tpl-' + this.dataName,true) :
26995             this.el.dom; 
26996         
26997         if(!node || node.parentNode == el){
26998                     return node;
26999             }
27000             var p = node.parentNode;
27001             while(p && p != el){
27002             if(p.parentNode == el){
27003                 return p;
27004             }
27005             p = p.parentNode;
27006         }
27007             return null;
27008     },
27009
27010     /** @ignore */
27011     onClick : function(e){
27012         var item = this.findItemFromChild(e.getTarget());
27013         if(item){
27014             var index = this.indexOf(item);
27015             if(this.onItemClick(item, index, e) !== false){
27016                 this.fireEvent("click", this, index, item, e);
27017             }
27018         }else{
27019             this.clearSelections();
27020         }
27021     },
27022
27023     /** @ignore */
27024     onContextMenu : function(e){
27025         var item = this.findItemFromChild(e.getTarget());
27026         if(item){
27027             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
27028         }
27029     },
27030
27031     /** @ignore */
27032     onDblClick : function(e){
27033         var item = this.findItemFromChild(e.getTarget());
27034         if(item){
27035             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
27036         }
27037     },
27038
27039     onItemClick : function(item, index, e)
27040     {
27041         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27042             return false;
27043         }
27044         if (this.toggleSelect) {
27045             var m = this.isSelected(item) ? 'unselect' : 'select';
27046             //Roo.log(m);
27047             var _t = this;
27048             _t[m](item, true, false);
27049             return true;
27050         }
27051         if(this.multiSelect || this.singleSelect){
27052             if(this.multiSelect && e.shiftKey && this.lastSelection){
27053                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27054             }else{
27055                 this.select(item, this.multiSelect && e.ctrlKey);
27056                 this.lastSelection = item;
27057             }
27058             
27059             if(!this.tickable){
27060                 e.preventDefault();
27061             }
27062             
27063         }
27064         return true;
27065     },
27066
27067     /**
27068      * Get the number of selected nodes.
27069      * @return {Number}
27070      */
27071     getSelectionCount : function(){
27072         return this.selections.length;
27073     },
27074
27075     /**
27076      * Get the currently selected nodes.
27077      * @return {Array} An array of HTMLElements
27078      */
27079     getSelectedNodes : function(){
27080         return this.selections;
27081     },
27082
27083     /**
27084      * Get the indexes of the selected nodes.
27085      * @return {Array}
27086      */
27087     getSelectedIndexes : function(){
27088         var indexes = [], s = this.selections;
27089         for(var i = 0, len = s.length; i < len; i++){
27090             indexes.push(s[i].nodeIndex);
27091         }
27092         return indexes;
27093     },
27094
27095     /**
27096      * Clear all selections
27097      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27098      */
27099     clearSelections : function(suppressEvent){
27100         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27101             this.cmp.elements = this.selections;
27102             this.cmp.removeClass(this.selectedClass);
27103             this.selections = [];
27104             if(!suppressEvent){
27105                 this.fireEvent("selectionchange", this, this.selections);
27106             }
27107         }
27108     },
27109
27110     /**
27111      * Returns true if the passed node is selected
27112      * @param {HTMLElement/Number} node The node or node index
27113      * @return {Boolean}
27114      */
27115     isSelected : function(node){
27116         var s = this.selections;
27117         if(s.length < 1){
27118             return false;
27119         }
27120         node = this.getNode(node);
27121         return s.indexOf(node) !== -1;
27122     },
27123
27124     /**
27125      * Selects nodes.
27126      * @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
27127      * @param {Boolean} keepExisting (optional) true to keep existing selections
27128      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27129      */
27130     select : function(nodeInfo, keepExisting, suppressEvent){
27131         if(nodeInfo instanceof Array){
27132             if(!keepExisting){
27133                 this.clearSelections(true);
27134             }
27135             for(var i = 0, len = nodeInfo.length; i < len; i++){
27136                 this.select(nodeInfo[i], true, true);
27137             }
27138             return;
27139         } 
27140         var node = this.getNode(nodeInfo);
27141         if(!node || this.isSelected(node)){
27142             return; // already selected.
27143         }
27144         if(!keepExisting){
27145             this.clearSelections(true);
27146         }
27147         
27148         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27149             Roo.fly(node).addClass(this.selectedClass);
27150             this.selections.push(node);
27151             if(!suppressEvent){
27152                 this.fireEvent("selectionchange", this, this.selections);
27153             }
27154         }
27155         
27156         
27157     },
27158       /**
27159      * Unselects nodes.
27160      * @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
27161      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27162      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27163      */
27164     unselect : function(nodeInfo, keepExisting, suppressEvent)
27165     {
27166         if(nodeInfo instanceof Array){
27167             Roo.each(this.selections, function(s) {
27168                 this.unselect(s, nodeInfo);
27169             }, this);
27170             return;
27171         }
27172         var node = this.getNode(nodeInfo);
27173         if(!node || !this.isSelected(node)){
27174             //Roo.log("not selected");
27175             return; // not selected.
27176         }
27177         // fireevent???
27178         var ns = [];
27179         Roo.each(this.selections, function(s) {
27180             if (s == node ) {
27181                 Roo.fly(node).removeClass(this.selectedClass);
27182
27183                 return;
27184             }
27185             ns.push(s);
27186         },this);
27187         
27188         this.selections= ns;
27189         this.fireEvent("selectionchange", this, this.selections);
27190     },
27191
27192     /**
27193      * Gets a template node.
27194      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27195      * @return {HTMLElement} The node or null if it wasn't found
27196      */
27197     getNode : function(nodeInfo){
27198         if(typeof nodeInfo == "string"){
27199             return document.getElementById(nodeInfo);
27200         }else if(typeof nodeInfo == "number"){
27201             return this.nodes[nodeInfo];
27202         }
27203         return nodeInfo;
27204     },
27205
27206     /**
27207      * Gets a range template nodes.
27208      * @param {Number} startIndex
27209      * @param {Number} endIndex
27210      * @return {Array} An array of nodes
27211      */
27212     getNodes : function(start, end){
27213         var ns = this.nodes;
27214         start = start || 0;
27215         end = typeof end == "undefined" ? ns.length - 1 : end;
27216         var nodes = [];
27217         if(start <= end){
27218             for(var i = start; i <= end; i++){
27219                 nodes.push(ns[i]);
27220             }
27221         } else{
27222             for(var i = start; i >= end; i--){
27223                 nodes.push(ns[i]);
27224             }
27225         }
27226         return nodes;
27227     },
27228
27229     /**
27230      * Finds the index of the passed node
27231      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27232      * @return {Number} The index of the node or -1
27233      */
27234     indexOf : function(node){
27235         node = this.getNode(node);
27236         if(typeof node.nodeIndex == "number"){
27237             return node.nodeIndex;
27238         }
27239         var ns = this.nodes;
27240         for(var i = 0, len = ns.length; i < len; i++){
27241             if(ns[i] == node){
27242                 return i;
27243             }
27244         }
27245         return -1;
27246     }
27247 });
27248 /*
27249  * Based on:
27250  * Ext JS Library 1.1.1
27251  * Copyright(c) 2006-2007, Ext JS, LLC.
27252  *
27253  * Originally Released Under LGPL - original licence link has changed is not relivant.
27254  *
27255  * Fork - LGPL
27256  * <script type="text/javascript">
27257  */
27258
27259 /**
27260  * @class Roo.JsonView
27261  * @extends Roo.View
27262  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27263 <pre><code>
27264 var view = new Roo.JsonView({
27265     container: "my-element",
27266     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27267     multiSelect: true, 
27268     jsonRoot: "data" 
27269 });
27270
27271 // listen for node click?
27272 view.on("click", function(vw, index, node, e){
27273     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27274 });
27275
27276 // direct load of JSON data
27277 view.load("foobar.php");
27278
27279 // Example from my blog list
27280 var tpl = new Roo.Template(
27281     '&lt;div class="entry"&gt;' +
27282     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27283     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27284     "&lt;/div&gt;&lt;hr /&gt;"
27285 );
27286
27287 var moreView = new Roo.JsonView({
27288     container :  "entry-list", 
27289     template : tpl,
27290     jsonRoot: "posts"
27291 });
27292 moreView.on("beforerender", this.sortEntries, this);
27293 moreView.load({
27294     url: "/blog/get-posts.php",
27295     params: "allposts=true",
27296     text: "Loading Blog Entries..."
27297 });
27298 </code></pre>
27299
27300 * Note: old code is supported with arguments : (container, template, config)
27301
27302
27303  * @constructor
27304  * Create a new JsonView
27305  * 
27306  * @param {Object} config The config object
27307  * 
27308  */
27309 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27310     
27311     
27312     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27313
27314     var um = this.el.getUpdateManager();
27315     um.setRenderer(this);
27316     um.on("update", this.onLoad, this);
27317     um.on("failure", this.onLoadException, this);
27318
27319     /**
27320      * @event beforerender
27321      * Fires before rendering of the downloaded JSON data.
27322      * @param {Roo.JsonView} this
27323      * @param {Object} data The JSON data loaded
27324      */
27325     /**
27326      * @event load
27327      * Fires when data is loaded.
27328      * @param {Roo.JsonView} this
27329      * @param {Object} data The JSON data loaded
27330      * @param {Object} response The raw Connect response object
27331      */
27332     /**
27333      * @event loadexception
27334      * Fires when loading fails.
27335      * @param {Roo.JsonView} this
27336      * @param {Object} response The raw Connect response object
27337      */
27338     this.addEvents({
27339         'beforerender' : true,
27340         'load' : true,
27341         'loadexception' : true
27342     });
27343 };
27344 Roo.extend(Roo.JsonView, Roo.View, {
27345     /**
27346      * @type {String} The root property in the loaded JSON object that contains the data
27347      */
27348     jsonRoot : "",
27349
27350     /**
27351      * Refreshes the view.
27352      */
27353     refresh : function(){
27354         this.clearSelections();
27355         this.el.update("");
27356         var html = [];
27357         var o = this.jsonData;
27358         if(o && o.length > 0){
27359             for(var i = 0, len = o.length; i < len; i++){
27360                 var data = this.prepareData(o[i], i, o);
27361                 html[html.length] = this.tpl.apply(data);
27362             }
27363         }else{
27364             html.push(this.emptyText);
27365         }
27366         this.el.update(html.join(""));
27367         this.nodes = this.el.dom.childNodes;
27368         this.updateIndexes(0);
27369     },
27370
27371     /**
27372      * 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.
27373      * @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:
27374      <pre><code>
27375      view.load({
27376          url: "your-url.php",
27377          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27378          callback: yourFunction,
27379          scope: yourObject, //(optional scope)
27380          discardUrl: false,
27381          nocache: false,
27382          text: "Loading...",
27383          timeout: 30,
27384          scripts: false
27385      });
27386      </code></pre>
27387      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27388      * 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.
27389      * @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}
27390      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27391      * @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.
27392      */
27393     load : function(){
27394         var um = this.el.getUpdateManager();
27395         um.update.apply(um, arguments);
27396     },
27397
27398     // note - render is a standard framework call...
27399     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27400     render : function(el, response){
27401         
27402         this.clearSelections();
27403         this.el.update("");
27404         var o;
27405         try{
27406             if (response != '') {
27407                 o = Roo.util.JSON.decode(response.responseText);
27408                 if(this.jsonRoot){
27409                     
27410                     o = o[this.jsonRoot];
27411                 }
27412             }
27413         } catch(e){
27414         }
27415         /**
27416          * The current JSON data or null
27417          */
27418         this.jsonData = o;
27419         this.beforeRender();
27420         this.refresh();
27421     },
27422
27423 /**
27424  * Get the number of records in the current JSON dataset
27425  * @return {Number}
27426  */
27427     getCount : function(){
27428         return this.jsonData ? this.jsonData.length : 0;
27429     },
27430
27431 /**
27432  * Returns the JSON object for the specified node(s)
27433  * @param {HTMLElement/Array} node The node or an array of nodes
27434  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27435  * you get the JSON object for the node
27436  */
27437     getNodeData : function(node){
27438         if(node instanceof Array){
27439             var data = [];
27440             for(var i = 0, len = node.length; i < len; i++){
27441                 data.push(this.getNodeData(node[i]));
27442             }
27443             return data;
27444         }
27445         return this.jsonData[this.indexOf(node)] || null;
27446     },
27447
27448     beforeRender : function(){
27449         this.snapshot = this.jsonData;
27450         if(this.sortInfo){
27451             this.sort.apply(this, this.sortInfo);
27452         }
27453         this.fireEvent("beforerender", this, this.jsonData);
27454     },
27455
27456     onLoad : function(el, o){
27457         this.fireEvent("load", this, this.jsonData, o);
27458     },
27459
27460     onLoadException : function(el, o){
27461         this.fireEvent("loadexception", this, o);
27462     },
27463
27464 /**
27465  * Filter the data by a specific property.
27466  * @param {String} property A property on your JSON objects
27467  * @param {String/RegExp} value Either string that the property values
27468  * should start with, or a RegExp to test against the property
27469  */
27470     filter : function(property, value){
27471         if(this.jsonData){
27472             var data = [];
27473             var ss = this.snapshot;
27474             if(typeof value == "string"){
27475                 var vlen = value.length;
27476                 if(vlen == 0){
27477                     this.clearFilter();
27478                     return;
27479                 }
27480                 value = value.toLowerCase();
27481                 for(var i = 0, len = ss.length; i < len; i++){
27482                     var o = ss[i];
27483                     if(o[property].substr(0, vlen).toLowerCase() == value){
27484                         data.push(o);
27485                     }
27486                 }
27487             } else if(value.exec){ // regex?
27488                 for(var i = 0, len = ss.length; i < len; i++){
27489                     var o = ss[i];
27490                     if(value.test(o[property])){
27491                         data.push(o);
27492                     }
27493                 }
27494             } else{
27495                 return;
27496             }
27497             this.jsonData = data;
27498             this.refresh();
27499         }
27500     },
27501
27502 /**
27503  * Filter by a function. The passed function will be called with each
27504  * object in the current dataset. If the function returns true the value is kept,
27505  * otherwise it is filtered.
27506  * @param {Function} fn
27507  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27508  */
27509     filterBy : function(fn, scope){
27510         if(this.jsonData){
27511             var data = [];
27512             var ss = this.snapshot;
27513             for(var i = 0, len = ss.length; i < len; i++){
27514                 var o = ss[i];
27515                 if(fn.call(scope || this, o)){
27516                     data.push(o);
27517                 }
27518             }
27519             this.jsonData = data;
27520             this.refresh();
27521         }
27522     },
27523
27524 /**
27525  * Clears the current filter.
27526  */
27527     clearFilter : function(){
27528         if(this.snapshot && this.jsonData != this.snapshot){
27529             this.jsonData = this.snapshot;
27530             this.refresh();
27531         }
27532     },
27533
27534
27535 /**
27536  * Sorts the data for this view and refreshes it.
27537  * @param {String} property A property on your JSON objects to sort on
27538  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27539  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27540  */
27541     sort : function(property, dir, sortType){
27542         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27543         if(this.jsonData){
27544             var p = property;
27545             var dsc = dir && dir.toLowerCase() == "desc";
27546             var f = function(o1, o2){
27547                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27548                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27549                 ;
27550                 if(v1 < v2){
27551                     return dsc ? +1 : -1;
27552                 } else if(v1 > v2){
27553                     return dsc ? -1 : +1;
27554                 } else{
27555                     return 0;
27556                 }
27557             };
27558             this.jsonData.sort(f);
27559             this.refresh();
27560             if(this.jsonData != this.snapshot){
27561                 this.snapshot.sort(f);
27562             }
27563         }
27564     }
27565 });/*
27566  * Based on:
27567  * Ext JS Library 1.1.1
27568  * Copyright(c) 2006-2007, Ext JS, LLC.
27569  *
27570  * Originally Released Under LGPL - original licence link has changed is not relivant.
27571  *
27572  * Fork - LGPL
27573  * <script type="text/javascript">
27574  */
27575  
27576
27577 /**
27578  * @class Roo.ColorPalette
27579  * @extends Roo.Component
27580  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27581  * Here's an example of typical usage:
27582  * <pre><code>
27583 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27584 cp.render('my-div');
27585
27586 cp.on('select', function(palette, selColor){
27587     // do something with selColor
27588 });
27589 </code></pre>
27590  * @constructor
27591  * Create a new ColorPalette
27592  * @param {Object} config The config object
27593  */
27594 Roo.ColorPalette = function(config){
27595     Roo.ColorPalette.superclass.constructor.call(this, config);
27596     this.addEvents({
27597         /**
27598              * @event select
27599              * Fires when a color is selected
27600              * @param {ColorPalette} this
27601              * @param {String} color The 6-digit color hex code (without the # symbol)
27602              */
27603         select: true
27604     });
27605
27606     if(this.handler){
27607         this.on("select", this.handler, this.scope, true);
27608     }
27609 };
27610 Roo.extend(Roo.ColorPalette, Roo.Component, {
27611     /**
27612      * @cfg {String} itemCls
27613      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27614      */
27615     itemCls : "x-color-palette",
27616     /**
27617      * @cfg {String} value
27618      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27619      * the hex codes are case-sensitive.
27620      */
27621     value : null,
27622     clickEvent:'click',
27623     // private
27624     ctype: "Roo.ColorPalette",
27625
27626     /**
27627      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27628      */
27629     allowReselect : false,
27630
27631     /**
27632      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27633      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27634      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27635      * of colors with the width setting until the box is symmetrical.</p>
27636      * <p>You can override individual colors if needed:</p>
27637      * <pre><code>
27638 var cp = new Roo.ColorPalette();
27639 cp.colors[0] = "FF0000";  // change the first box to red
27640 </code></pre>
27641
27642 Or you can provide a custom array of your own for complete control:
27643 <pre><code>
27644 var cp = new Roo.ColorPalette();
27645 cp.colors = ["000000", "993300", "333300"];
27646 </code></pre>
27647      * @type Array
27648      */
27649     colors : [
27650         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27651         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27652         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27653         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27654         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27655     ],
27656
27657     // private
27658     onRender : function(container, position){
27659         var t = new Roo.MasterTemplate(
27660             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27661         );
27662         var c = this.colors;
27663         for(var i = 0, len = c.length; i < len; i++){
27664             t.add([c[i]]);
27665         }
27666         var el = document.createElement("div");
27667         el.className = this.itemCls;
27668         t.overwrite(el);
27669         container.dom.insertBefore(el, position);
27670         this.el = Roo.get(el);
27671         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27672         if(this.clickEvent != 'click'){
27673             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27674         }
27675     },
27676
27677     // private
27678     afterRender : function(){
27679         Roo.ColorPalette.superclass.afterRender.call(this);
27680         if(this.value){
27681             var s = this.value;
27682             this.value = null;
27683             this.select(s);
27684         }
27685     },
27686
27687     // private
27688     handleClick : function(e, t){
27689         e.preventDefault();
27690         if(!this.disabled){
27691             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27692             this.select(c.toUpperCase());
27693         }
27694     },
27695
27696     /**
27697      * Selects the specified color in the palette (fires the select event)
27698      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27699      */
27700     select : function(color){
27701         color = color.replace("#", "");
27702         if(color != this.value || this.allowReselect){
27703             var el = this.el;
27704             if(this.value){
27705                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27706             }
27707             el.child("a.color-"+color).addClass("x-color-palette-sel");
27708             this.value = color;
27709             this.fireEvent("select", this, color);
27710         }
27711     }
27712 });/*
27713  * Based on:
27714  * Ext JS Library 1.1.1
27715  * Copyright(c) 2006-2007, Ext JS, LLC.
27716  *
27717  * Originally Released Under LGPL - original licence link has changed is not relivant.
27718  *
27719  * Fork - LGPL
27720  * <script type="text/javascript">
27721  */
27722  
27723 /**
27724  * @class Roo.DatePicker
27725  * @extends Roo.Component
27726  * Simple date picker class.
27727  * @constructor
27728  * Create a new DatePicker
27729  * @param {Object} config The config object
27730  */
27731 Roo.DatePicker = function(config){
27732     Roo.DatePicker.superclass.constructor.call(this, config);
27733
27734     this.value = config && config.value ?
27735                  config.value.clearTime() : new Date().clearTime();
27736
27737     this.addEvents({
27738         /**
27739              * @event select
27740              * Fires when a date is selected
27741              * @param {DatePicker} this
27742              * @param {Date} date The selected date
27743              */
27744         'select': true,
27745         /**
27746              * @event monthchange
27747              * Fires when the displayed month changes 
27748              * @param {DatePicker} this
27749              * @param {Date} date The selected month
27750              */
27751         'monthchange': true
27752     });
27753
27754     if(this.handler){
27755         this.on("select", this.handler,  this.scope || this);
27756     }
27757     // build the disabledDatesRE
27758     if(!this.disabledDatesRE && this.disabledDates){
27759         var dd = this.disabledDates;
27760         var re = "(?:";
27761         for(var i = 0; i < dd.length; i++){
27762             re += dd[i];
27763             if(i != dd.length-1) {
27764                 re += "|";
27765             }
27766         }
27767         this.disabledDatesRE = new RegExp(re + ")");
27768     }
27769 };
27770
27771 Roo.extend(Roo.DatePicker, Roo.Component, {
27772     /**
27773      * @cfg {String} todayText
27774      * The text to display on the button that selects the current date (defaults to "Today")
27775      */
27776     todayText : "Today",
27777     /**
27778      * @cfg {String} okText
27779      * The text to display on the ok button
27780      */
27781     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27782     /**
27783      * @cfg {String} cancelText
27784      * The text to display on the cancel button
27785      */
27786     cancelText : "Cancel",
27787     /**
27788      * @cfg {String} todayTip
27789      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27790      */
27791     todayTip : "{0} (Spacebar)",
27792     /**
27793      * @cfg {Date} minDate
27794      * Minimum allowable date (JavaScript date object, defaults to null)
27795      */
27796     minDate : null,
27797     /**
27798      * @cfg {Date} maxDate
27799      * Maximum allowable date (JavaScript date object, defaults to null)
27800      */
27801     maxDate : null,
27802     /**
27803      * @cfg {String} minText
27804      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27805      */
27806     minText : "This date is before the minimum date",
27807     /**
27808      * @cfg {String} maxText
27809      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27810      */
27811     maxText : "This date is after the maximum date",
27812     /**
27813      * @cfg {String} format
27814      * The default date format string which can be overriden for localization support.  The format must be
27815      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27816      */
27817     format : "m/d/y",
27818     /**
27819      * @cfg {Array} disabledDays
27820      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27821      */
27822     disabledDays : null,
27823     /**
27824      * @cfg {String} disabledDaysText
27825      * The tooltip to display when the date falls on a disabled day (defaults to "")
27826      */
27827     disabledDaysText : "",
27828     /**
27829      * @cfg {RegExp} disabledDatesRE
27830      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27831      */
27832     disabledDatesRE : null,
27833     /**
27834      * @cfg {String} disabledDatesText
27835      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27836      */
27837     disabledDatesText : "",
27838     /**
27839      * @cfg {Boolean} constrainToViewport
27840      * True to constrain the date picker to the viewport (defaults to true)
27841      */
27842     constrainToViewport : true,
27843     /**
27844      * @cfg {Array} monthNames
27845      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27846      */
27847     monthNames : Date.monthNames,
27848     /**
27849      * @cfg {Array} dayNames
27850      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27851      */
27852     dayNames : Date.dayNames,
27853     /**
27854      * @cfg {String} nextText
27855      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27856      */
27857     nextText: 'Next Month (Control+Right)',
27858     /**
27859      * @cfg {String} prevText
27860      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27861      */
27862     prevText: 'Previous Month (Control+Left)',
27863     /**
27864      * @cfg {String} monthYearText
27865      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27866      */
27867     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27868     /**
27869      * @cfg {Number} startDay
27870      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27871      */
27872     startDay : 0,
27873     /**
27874      * @cfg {Bool} showClear
27875      * Show a clear button (usefull for date form elements that can be blank.)
27876      */
27877     
27878     showClear: false,
27879     
27880     /**
27881      * Sets the value of the date field
27882      * @param {Date} value The date to set
27883      */
27884     setValue : function(value){
27885         var old = this.value;
27886         
27887         if (typeof(value) == 'string') {
27888          
27889             value = Date.parseDate(value, this.format);
27890         }
27891         if (!value) {
27892             value = new Date();
27893         }
27894         
27895         this.value = value.clearTime(true);
27896         if(this.el){
27897             this.update(this.value);
27898         }
27899     },
27900
27901     /**
27902      * Gets the current selected value of the date field
27903      * @return {Date} The selected date
27904      */
27905     getValue : function(){
27906         return this.value;
27907     },
27908
27909     // private
27910     focus : function(){
27911         if(this.el){
27912             this.update(this.activeDate);
27913         }
27914     },
27915
27916     // privateval
27917     onRender : function(container, position){
27918         
27919         var m = [
27920              '<table cellspacing="0">',
27921                 '<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>',
27922                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27923         var dn = this.dayNames;
27924         for(var i = 0; i < 7; i++){
27925             var d = this.startDay+i;
27926             if(d > 6){
27927                 d = d-7;
27928             }
27929             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27930         }
27931         m[m.length] = "</tr></thead><tbody><tr>";
27932         for(var i = 0; i < 42; i++) {
27933             if(i % 7 == 0 && i != 0){
27934                 m[m.length] = "</tr><tr>";
27935             }
27936             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27937         }
27938         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27939             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27940
27941         var el = document.createElement("div");
27942         el.className = "x-date-picker";
27943         el.innerHTML = m.join("");
27944
27945         container.dom.insertBefore(el, position);
27946
27947         this.el = Roo.get(el);
27948         this.eventEl = Roo.get(el.firstChild);
27949
27950         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27951             handler: this.showPrevMonth,
27952             scope: this,
27953             preventDefault:true,
27954             stopDefault:true
27955         });
27956
27957         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27958             handler: this.showNextMonth,
27959             scope: this,
27960             preventDefault:true,
27961             stopDefault:true
27962         });
27963
27964         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27965
27966         this.monthPicker = this.el.down('div.x-date-mp');
27967         this.monthPicker.enableDisplayMode('block');
27968         
27969         var kn = new Roo.KeyNav(this.eventEl, {
27970             "left" : function(e){
27971                 e.ctrlKey ?
27972                     this.showPrevMonth() :
27973                     this.update(this.activeDate.add("d", -1));
27974             },
27975
27976             "right" : function(e){
27977                 e.ctrlKey ?
27978                     this.showNextMonth() :
27979                     this.update(this.activeDate.add("d", 1));
27980             },
27981
27982             "up" : function(e){
27983                 e.ctrlKey ?
27984                     this.showNextYear() :
27985                     this.update(this.activeDate.add("d", -7));
27986             },
27987
27988             "down" : function(e){
27989                 e.ctrlKey ?
27990                     this.showPrevYear() :
27991                     this.update(this.activeDate.add("d", 7));
27992             },
27993
27994             "pageUp" : function(e){
27995                 this.showNextMonth();
27996             },
27997
27998             "pageDown" : function(e){
27999                 this.showPrevMonth();
28000             },
28001
28002             "enter" : function(e){
28003                 e.stopPropagation();
28004                 return true;
28005             },
28006
28007             scope : this
28008         });
28009
28010         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
28011
28012         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
28013
28014         this.el.unselectable();
28015         
28016         this.cells = this.el.select("table.x-date-inner tbody td");
28017         this.textNodes = this.el.query("table.x-date-inner tbody span");
28018
28019         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
28020             text: "&#160;",
28021             tooltip: this.monthYearText
28022         });
28023
28024         this.mbtn.on('click', this.showMonthPicker, this);
28025         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
28026
28027
28028         var today = (new Date()).dateFormat(this.format);
28029         
28030         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
28031         if (this.showClear) {
28032             baseTb.add( new Roo.Toolbar.Fill());
28033         }
28034         baseTb.add({
28035             text: String.format(this.todayText, today),
28036             tooltip: String.format(this.todayTip, today),
28037             handler: this.selectToday,
28038             scope: this
28039         });
28040         
28041         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28042             
28043         //});
28044         if (this.showClear) {
28045             
28046             baseTb.add( new Roo.Toolbar.Fill());
28047             baseTb.add({
28048                 text: '&#160;',
28049                 cls: 'x-btn-icon x-btn-clear',
28050                 handler: function() {
28051                     //this.value = '';
28052                     this.fireEvent("select", this, '');
28053                 },
28054                 scope: this
28055             });
28056         }
28057         
28058         
28059         if(Roo.isIE){
28060             this.el.repaint();
28061         }
28062         this.update(this.value);
28063     },
28064
28065     createMonthPicker : function(){
28066         if(!this.monthPicker.dom.firstChild){
28067             var buf = ['<table border="0" cellspacing="0">'];
28068             for(var i = 0; i < 6; i++){
28069                 buf.push(
28070                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28071                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28072                     i == 0 ?
28073                     '<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>' :
28074                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28075                 );
28076             }
28077             buf.push(
28078                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28079                     this.okText,
28080                     '</button><button type="button" class="x-date-mp-cancel">',
28081                     this.cancelText,
28082                     '</button></td></tr>',
28083                 '</table>'
28084             );
28085             this.monthPicker.update(buf.join(''));
28086             this.monthPicker.on('click', this.onMonthClick, this);
28087             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28088
28089             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28090             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28091
28092             this.mpMonths.each(function(m, a, i){
28093                 i += 1;
28094                 if((i%2) == 0){
28095                     m.dom.xmonth = 5 + Math.round(i * .5);
28096                 }else{
28097                     m.dom.xmonth = Math.round((i-1) * .5);
28098                 }
28099             });
28100         }
28101     },
28102
28103     showMonthPicker : function(){
28104         this.createMonthPicker();
28105         var size = this.el.getSize();
28106         this.monthPicker.setSize(size);
28107         this.monthPicker.child('table').setSize(size);
28108
28109         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28110         this.updateMPMonth(this.mpSelMonth);
28111         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28112         this.updateMPYear(this.mpSelYear);
28113
28114         this.monthPicker.slideIn('t', {duration:.2});
28115     },
28116
28117     updateMPYear : function(y){
28118         this.mpyear = y;
28119         var ys = this.mpYears.elements;
28120         for(var i = 1; i <= 10; i++){
28121             var td = ys[i-1], y2;
28122             if((i%2) == 0){
28123                 y2 = y + Math.round(i * .5);
28124                 td.firstChild.innerHTML = y2;
28125                 td.xyear = y2;
28126             }else{
28127                 y2 = y - (5-Math.round(i * .5));
28128                 td.firstChild.innerHTML = y2;
28129                 td.xyear = y2;
28130             }
28131             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28132         }
28133     },
28134
28135     updateMPMonth : function(sm){
28136         this.mpMonths.each(function(m, a, i){
28137             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28138         });
28139     },
28140
28141     selectMPMonth: function(m){
28142         
28143     },
28144
28145     onMonthClick : function(e, t){
28146         e.stopEvent();
28147         var el = new Roo.Element(t), pn;
28148         if(el.is('button.x-date-mp-cancel')){
28149             this.hideMonthPicker();
28150         }
28151         else if(el.is('button.x-date-mp-ok')){
28152             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28153             this.hideMonthPicker();
28154         }
28155         else if(pn = el.up('td.x-date-mp-month', 2)){
28156             this.mpMonths.removeClass('x-date-mp-sel');
28157             pn.addClass('x-date-mp-sel');
28158             this.mpSelMonth = pn.dom.xmonth;
28159         }
28160         else if(pn = el.up('td.x-date-mp-year', 2)){
28161             this.mpYears.removeClass('x-date-mp-sel');
28162             pn.addClass('x-date-mp-sel');
28163             this.mpSelYear = pn.dom.xyear;
28164         }
28165         else if(el.is('a.x-date-mp-prev')){
28166             this.updateMPYear(this.mpyear-10);
28167         }
28168         else if(el.is('a.x-date-mp-next')){
28169             this.updateMPYear(this.mpyear+10);
28170         }
28171     },
28172
28173     onMonthDblClick : function(e, t){
28174         e.stopEvent();
28175         var el = new Roo.Element(t), pn;
28176         if(pn = el.up('td.x-date-mp-month', 2)){
28177             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28178             this.hideMonthPicker();
28179         }
28180         else if(pn = el.up('td.x-date-mp-year', 2)){
28181             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28182             this.hideMonthPicker();
28183         }
28184     },
28185
28186     hideMonthPicker : function(disableAnim){
28187         if(this.monthPicker){
28188             if(disableAnim === true){
28189                 this.monthPicker.hide();
28190             }else{
28191                 this.monthPicker.slideOut('t', {duration:.2});
28192             }
28193         }
28194     },
28195
28196     // private
28197     showPrevMonth : function(e){
28198         this.update(this.activeDate.add("mo", -1));
28199     },
28200
28201     // private
28202     showNextMonth : function(e){
28203         this.update(this.activeDate.add("mo", 1));
28204     },
28205
28206     // private
28207     showPrevYear : function(){
28208         this.update(this.activeDate.add("y", -1));
28209     },
28210
28211     // private
28212     showNextYear : function(){
28213         this.update(this.activeDate.add("y", 1));
28214     },
28215
28216     // private
28217     handleMouseWheel : function(e){
28218         var delta = e.getWheelDelta();
28219         if(delta > 0){
28220             this.showPrevMonth();
28221             e.stopEvent();
28222         } else if(delta < 0){
28223             this.showNextMonth();
28224             e.stopEvent();
28225         }
28226     },
28227
28228     // private
28229     handleDateClick : function(e, t){
28230         e.stopEvent();
28231         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28232             this.setValue(new Date(t.dateValue));
28233             this.fireEvent("select", this, this.value);
28234         }
28235     },
28236
28237     // private
28238     selectToday : function(){
28239         this.setValue(new Date().clearTime());
28240         this.fireEvent("select", this, this.value);
28241     },
28242
28243     // private
28244     update : function(date)
28245     {
28246         var vd = this.activeDate;
28247         this.activeDate = date;
28248         if(vd && this.el){
28249             var t = date.getTime();
28250             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28251                 this.cells.removeClass("x-date-selected");
28252                 this.cells.each(function(c){
28253                    if(c.dom.firstChild.dateValue == t){
28254                        c.addClass("x-date-selected");
28255                        setTimeout(function(){
28256                             try{c.dom.firstChild.focus();}catch(e){}
28257                        }, 50);
28258                        return false;
28259                    }
28260                 });
28261                 return;
28262             }
28263         }
28264         
28265         var days = date.getDaysInMonth();
28266         var firstOfMonth = date.getFirstDateOfMonth();
28267         var startingPos = firstOfMonth.getDay()-this.startDay;
28268
28269         if(startingPos <= this.startDay){
28270             startingPos += 7;
28271         }
28272
28273         var pm = date.add("mo", -1);
28274         var prevStart = pm.getDaysInMonth()-startingPos;
28275
28276         var cells = this.cells.elements;
28277         var textEls = this.textNodes;
28278         days += startingPos;
28279
28280         // convert everything to numbers so it's fast
28281         var day = 86400000;
28282         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28283         var today = new Date().clearTime().getTime();
28284         var sel = date.clearTime().getTime();
28285         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28286         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28287         var ddMatch = this.disabledDatesRE;
28288         var ddText = this.disabledDatesText;
28289         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28290         var ddaysText = this.disabledDaysText;
28291         var format = this.format;
28292
28293         var setCellClass = function(cal, cell){
28294             cell.title = "";
28295             var t = d.getTime();
28296             cell.firstChild.dateValue = t;
28297             if(t == today){
28298                 cell.className += " x-date-today";
28299                 cell.title = cal.todayText;
28300             }
28301             if(t == sel){
28302                 cell.className += " x-date-selected";
28303                 setTimeout(function(){
28304                     try{cell.firstChild.focus();}catch(e){}
28305                 }, 50);
28306             }
28307             // disabling
28308             if(t < min) {
28309                 cell.className = " x-date-disabled";
28310                 cell.title = cal.minText;
28311                 return;
28312             }
28313             if(t > max) {
28314                 cell.className = " x-date-disabled";
28315                 cell.title = cal.maxText;
28316                 return;
28317             }
28318             if(ddays){
28319                 if(ddays.indexOf(d.getDay()) != -1){
28320                     cell.title = ddaysText;
28321                     cell.className = " x-date-disabled";
28322                 }
28323             }
28324             if(ddMatch && format){
28325                 var fvalue = d.dateFormat(format);
28326                 if(ddMatch.test(fvalue)){
28327                     cell.title = ddText.replace("%0", fvalue);
28328                     cell.className = " x-date-disabled";
28329                 }
28330             }
28331         };
28332
28333         var i = 0;
28334         for(; i < startingPos; i++) {
28335             textEls[i].innerHTML = (++prevStart);
28336             d.setDate(d.getDate()+1);
28337             cells[i].className = "x-date-prevday";
28338             setCellClass(this, cells[i]);
28339         }
28340         for(; i < days; i++){
28341             intDay = i - startingPos + 1;
28342             textEls[i].innerHTML = (intDay);
28343             d.setDate(d.getDate()+1);
28344             cells[i].className = "x-date-active";
28345             setCellClass(this, cells[i]);
28346         }
28347         var extraDays = 0;
28348         for(; i < 42; i++) {
28349              textEls[i].innerHTML = (++extraDays);
28350              d.setDate(d.getDate()+1);
28351              cells[i].className = "x-date-nextday";
28352              setCellClass(this, cells[i]);
28353         }
28354
28355         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28356         this.fireEvent('monthchange', this, date);
28357         
28358         if(!this.internalRender){
28359             var main = this.el.dom.firstChild;
28360             var w = main.offsetWidth;
28361             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28362             Roo.fly(main).setWidth(w);
28363             this.internalRender = true;
28364             // opera does not respect the auto grow header center column
28365             // then, after it gets a width opera refuses to recalculate
28366             // without a second pass
28367             if(Roo.isOpera && !this.secondPass){
28368                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28369                 this.secondPass = true;
28370                 this.update.defer(10, this, [date]);
28371             }
28372         }
28373         
28374         
28375     }
28376 });        /*
28377  * Based on:
28378  * Ext JS Library 1.1.1
28379  * Copyright(c) 2006-2007, Ext JS, LLC.
28380  *
28381  * Originally Released Under LGPL - original licence link has changed is not relivant.
28382  *
28383  * Fork - LGPL
28384  * <script type="text/javascript">
28385  */
28386 /**
28387  * @class Roo.TabPanel
28388  * @extends Roo.util.Observable
28389  * A lightweight tab container.
28390  * <br><br>
28391  * Usage:
28392  * <pre><code>
28393 // basic tabs 1, built from existing content
28394 var tabs = new Roo.TabPanel("tabs1");
28395 tabs.addTab("script", "View Script");
28396 tabs.addTab("markup", "View Markup");
28397 tabs.activate("script");
28398
28399 // more advanced tabs, built from javascript
28400 var jtabs = new Roo.TabPanel("jtabs");
28401 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28402
28403 // set up the UpdateManager
28404 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28405 var updater = tab2.getUpdateManager();
28406 updater.setDefaultUrl("ajax1.htm");
28407 tab2.on('activate', updater.refresh, updater, true);
28408
28409 // Use setUrl for Ajax loading
28410 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28411 tab3.setUrl("ajax2.htm", null, true);
28412
28413 // Disabled tab
28414 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28415 tab4.disable();
28416
28417 jtabs.activate("jtabs-1");
28418  * </code></pre>
28419  * @constructor
28420  * Create a new TabPanel.
28421  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28422  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28423  */
28424 Roo.TabPanel = function(container, config){
28425     /**
28426     * The container element for this TabPanel.
28427     * @type Roo.Element
28428     */
28429     this.el = Roo.get(container, true);
28430     if(config){
28431         if(typeof config == "boolean"){
28432             this.tabPosition = config ? "bottom" : "top";
28433         }else{
28434             Roo.apply(this, config);
28435         }
28436     }
28437     if(this.tabPosition == "bottom"){
28438         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28439         this.el.addClass("x-tabs-bottom");
28440     }
28441     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28442     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28443     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28444     if(Roo.isIE){
28445         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28446     }
28447     if(this.tabPosition != "bottom"){
28448         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28449          * @type Roo.Element
28450          */
28451         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28452         this.el.addClass("x-tabs-top");
28453     }
28454     this.items = [];
28455
28456     this.bodyEl.setStyle("position", "relative");
28457
28458     this.active = null;
28459     this.activateDelegate = this.activate.createDelegate(this);
28460
28461     this.addEvents({
28462         /**
28463          * @event tabchange
28464          * Fires when the active tab changes
28465          * @param {Roo.TabPanel} this
28466          * @param {Roo.TabPanelItem} activePanel The new active tab
28467          */
28468         "tabchange": true,
28469         /**
28470          * @event beforetabchange
28471          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28472          * @param {Roo.TabPanel} this
28473          * @param {Object} e Set cancel to true on this object to cancel the tab change
28474          * @param {Roo.TabPanelItem} tab The tab being changed to
28475          */
28476         "beforetabchange" : true
28477     });
28478
28479     Roo.EventManager.onWindowResize(this.onResize, this);
28480     this.cpad = this.el.getPadding("lr");
28481     this.hiddenCount = 0;
28482
28483
28484     // toolbar on the tabbar support...
28485     if (this.toolbar) {
28486         var tcfg = this.toolbar;
28487         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28488         this.toolbar = new Roo.Toolbar(tcfg);
28489         if (Roo.isSafari) {
28490             var tbl = tcfg.container.child('table', true);
28491             tbl.setAttribute('width', '100%');
28492         }
28493         
28494     }
28495    
28496
28497
28498     Roo.TabPanel.superclass.constructor.call(this);
28499 };
28500
28501 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28502     /*
28503      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28504      */
28505     tabPosition : "top",
28506     /*
28507      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28508      */
28509     currentTabWidth : 0,
28510     /*
28511      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28512      */
28513     minTabWidth : 40,
28514     /*
28515      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28516      */
28517     maxTabWidth : 250,
28518     /*
28519      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28520      */
28521     preferredTabWidth : 175,
28522     /*
28523      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28524      */
28525     resizeTabs : false,
28526     /*
28527      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28528      */
28529     monitorResize : true,
28530     /*
28531      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28532      */
28533     toolbar : false,
28534
28535     /**
28536      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28537      * @param {String} id The id of the div to use <b>or create</b>
28538      * @param {String} text The text for the tab
28539      * @param {String} content (optional) Content to put in the TabPanelItem body
28540      * @param {Boolean} closable (optional) True to create a close icon on the tab
28541      * @return {Roo.TabPanelItem} The created TabPanelItem
28542      */
28543     addTab : function(id, text, content, closable){
28544         var item = new Roo.TabPanelItem(this, id, text, closable);
28545         this.addTabItem(item);
28546         if(content){
28547             item.setContent(content);
28548         }
28549         return item;
28550     },
28551
28552     /**
28553      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28554      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28555      * @return {Roo.TabPanelItem}
28556      */
28557     getTab : function(id){
28558         return this.items[id];
28559     },
28560
28561     /**
28562      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28563      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28564      */
28565     hideTab : function(id){
28566         var t = this.items[id];
28567         if(!t.isHidden()){
28568            t.setHidden(true);
28569            this.hiddenCount++;
28570            this.autoSizeTabs();
28571         }
28572     },
28573
28574     /**
28575      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28576      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28577      */
28578     unhideTab : function(id){
28579         var t = this.items[id];
28580         if(t.isHidden()){
28581            t.setHidden(false);
28582            this.hiddenCount--;
28583            this.autoSizeTabs();
28584         }
28585     },
28586
28587     /**
28588      * Adds an existing {@link Roo.TabPanelItem}.
28589      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28590      */
28591     addTabItem : function(item){
28592         this.items[item.id] = item;
28593         this.items.push(item);
28594         if(this.resizeTabs){
28595            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28596            this.autoSizeTabs();
28597         }else{
28598             item.autoSize();
28599         }
28600     },
28601
28602     /**
28603      * Removes a {@link Roo.TabPanelItem}.
28604      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28605      */
28606     removeTab : function(id){
28607         var items = this.items;
28608         var tab = items[id];
28609         if(!tab) { return; }
28610         var index = items.indexOf(tab);
28611         if(this.active == tab && items.length > 1){
28612             var newTab = this.getNextAvailable(index);
28613             if(newTab) {
28614                 newTab.activate();
28615             }
28616         }
28617         this.stripEl.dom.removeChild(tab.pnode.dom);
28618         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28619             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28620         }
28621         items.splice(index, 1);
28622         delete this.items[tab.id];
28623         tab.fireEvent("close", tab);
28624         tab.purgeListeners();
28625         this.autoSizeTabs();
28626     },
28627
28628     getNextAvailable : function(start){
28629         var items = this.items;
28630         var index = start;
28631         // look for a next tab that will slide over to
28632         // replace the one being removed
28633         while(index < items.length){
28634             var item = items[++index];
28635             if(item && !item.isHidden()){
28636                 return item;
28637             }
28638         }
28639         // if one isn't found select the previous tab (on the left)
28640         index = start;
28641         while(index >= 0){
28642             var item = items[--index];
28643             if(item && !item.isHidden()){
28644                 return item;
28645             }
28646         }
28647         return null;
28648     },
28649
28650     /**
28651      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28652      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28653      */
28654     disableTab : function(id){
28655         var tab = this.items[id];
28656         if(tab && this.active != tab){
28657             tab.disable();
28658         }
28659     },
28660
28661     /**
28662      * Enables a {@link Roo.TabPanelItem} that is disabled.
28663      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28664      */
28665     enableTab : function(id){
28666         var tab = this.items[id];
28667         tab.enable();
28668     },
28669
28670     /**
28671      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28672      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28673      * @return {Roo.TabPanelItem} The TabPanelItem.
28674      */
28675     activate : function(id){
28676         var tab = this.items[id];
28677         if(!tab){
28678             return null;
28679         }
28680         if(tab == this.active || tab.disabled){
28681             return tab;
28682         }
28683         var e = {};
28684         this.fireEvent("beforetabchange", this, e, tab);
28685         if(e.cancel !== true && !tab.disabled){
28686             if(this.active){
28687                 this.active.hide();
28688             }
28689             this.active = this.items[id];
28690             this.active.show();
28691             this.fireEvent("tabchange", this, this.active);
28692         }
28693         return tab;
28694     },
28695
28696     /**
28697      * Gets the active {@link Roo.TabPanelItem}.
28698      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28699      */
28700     getActiveTab : function(){
28701         return this.active;
28702     },
28703
28704     /**
28705      * Updates the tab body element to fit the height of the container element
28706      * for overflow scrolling
28707      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28708      */
28709     syncHeight : function(targetHeight){
28710         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28711         var bm = this.bodyEl.getMargins();
28712         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28713         this.bodyEl.setHeight(newHeight);
28714         return newHeight;
28715     },
28716
28717     onResize : function(){
28718         if(this.monitorResize){
28719             this.autoSizeTabs();
28720         }
28721     },
28722
28723     /**
28724      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28725      */
28726     beginUpdate : function(){
28727         this.updating = true;
28728     },
28729
28730     /**
28731      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28732      */
28733     endUpdate : function(){
28734         this.updating = false;
28735         this.autoSizeTabs();
28736     },
28737
28738     /**
28739      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28740      */
28741     autoSizeTabs : function(){
28742         var count = this.items.length;
28743         var vcount = count - this.hiddenCount;
28744         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28745             return;
28746         }
28747         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28748         var availWidth = Math.floor(w / vcount);
28749         var b = this.stripBody;
28750         if(b.getWidth() > w){
28751             var tabs = this.items;
28752             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28753             if(availWidth < this.minTabWidth){
28754                 /*if(!this.sleft){    // incomplete scrolling code
28755                     this.createScrollButtons();
28756                 }
28757                 this.showScroll();
28758                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28759             }
28760         }else{
28761             if(this.currentTabWidth < this.preferredTabWidth){
28762                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28763             }
28764         }
28765     },
28766
28767     /**
28768      * Returns the number of tabs in this TabPanel.
28769      * @return {Number}
28770      */
28771      getCount : function(){
28772          return this.items.length;
28773      },
28774
28775     /**
28776      * Resizes all the tabs to the passed width
28777      * @param {Number} The new width
28778      */
28779     setTabWidth : function(width){
28780         this.currentTabWidth = width;
28781         for(var i = 0, len = this.items.length; i < len; i++) {
28782                 if(!this.items[i].isHidden()) {
28783                 this.items[i].setWidth(width);
28784             }
28785         }
28786     },
28787
28788     /**
28789      * Destroys this TabPanel
28790      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28791      */
28792     destroy : function(removeEl){
28793         Roo.EventManager.removeResizeListener(this.onResize, this);
28794         for(var i = 0, len = this.items.length; i < len; i++){
28795             this.items[i].purgeListeners();
28796         }
28797         if(removeEl === true){
28798             this.el.update("");
28799             this.el.remove();
28800         }
28801     }
28802 });
28803
28804 /**
28805  * @class Roo.TabPanelItem
28806  * @extends Roo.util.Observable
28807  * Represents an individual item (tab plus body) in a TabPanel.
28808  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28809  * @param {String} id The id of this TabPanelItem
28810  * @param {String} text The text for the tab of this TabPanelItem
28811  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28812  */
28813 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28814     /**
28815      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28816      * @type Roo.TabPanel
28817      */
28818     this.tabPanel = tabPanel;
28819     /**
28820      * The id for this TabPanelItem
28821      * @type String
28822      */
28823     this.id = id;
28824     /** @private */
28825     this.disabled = false;
28826     /** @private */
28827     this.text = text;
28828     /** @private */
28829     this.loaded = false;
28830     this.closable = closable;
28831
28832     /**
28833      * The body element for this TabPanelItem.
28834      * @type Roo.Element
28835      */
28836     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28837     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28838     this.bodyEl.setStyle("display", "block");
28839     this.bodyEl.setStyle("zoom", "1");
28840     this.hideAction();
28841
28842     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28843     /** @private */
28844     this.el = Roo.get(els.el, true);
28845     this.inner = Roo.get(els.inner, true);
28846     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28847     this.pnode = Roo.get(els.el.parentNode, true);
28848     this.el.on("mousedown", this.onTabMouseDown, this);
28849     this.el.on("click", this.onTabClick, this);
28850     /** @private */
28851     if(closable){
28852         var c = Roo.get(els.close, true);
28853         c.dom.title = this.closeText;
28854         c.addClassOnOver("close-over");
28855         c.on("click", this.closeClick, this);
28856      }
28857
28858     this.addEvents({
28859          /**
28860          * @event activate
28861          * Fires when this tab becomes the active tab.
28862          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28863          * @param {Roo.TabPanelItem} this
28864          */
28865         "activate": true,
28866         /**
28867          * @event beforeclose
28868          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28869          * @param {Roo.TabPanelItem} this
28870          * @param {Object} e Set cancel to true on this object to cancel the close.
28871          */
28872         "beforeclose": true,
28873         /**
28874          * @event close
28875          * Fires when this tab is closed.
28876          * @param {Roo.TabPanelItem} this
28877          */
28878          "close": true,
28879         /**
28880          * @event deactivate
28881          * Fires when this tab is no longer the active tab.
28882          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28883          * @param {Roo.TabPanelItem} this
28884          */
28885          "deactivate" : true
28886     });
28887     this.hidden = false;
28888
28889     Roo.TabPanelItem.superclass.constructor.call(this);
28890 };
28891
28892 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28893     purgeListeners : function(){
28894        Roo.util.Observable.prototype.purgeListeners.call(this);
28895        this.el.removeAllListeners();
28896     },
28897     /**
28898      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28899      */
28900     show : function(){
28901         this.pnode.addClass("on");
28902         this.showAction();
28903         if(Roo.isOpera){
28904             this.tabPanel.stripWrap.repaint();
28905         }
28906         this.fireEvent("activate", this.tabPanel, this);
28907     },
28908
28909     /**
28910      * Returns true if this tab is the active tab.
28911      * @return {Boolean}
28912      */
28913     isActive : function(){
28914         return this.tabPanel.getActiveTab() == this;
28915     },
28916
28917     /**
28918      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28919      */
28920     hide : function(){
28921         this.pnode.removeClass("on");
28922         this.hideAction();
28923         this.fireEvent("deactivate", this.tabPanel, this);
28924     },
28925
28926     hideAction : function(){
28927         this.bodyEl.hide();
28928         this.bodyEl.setStyle("position", "absolute");
28929         this.bodyEl.setLeft("-20000px");
28930         this.bodyEl.setTop("-20000px");
28931     },
28932
28933     showAction : function(){
28934         this.bodyEl.setStyle("position", "relative");
28935         this.bodyEl.setTop("");
28936         this.bodyEl.setLeft("");
28937         this.bodyEl.show();
28938     },
28939
28940     /**
28941      * Set the tooltip for the tab.
28942      * @param {String} tooltip The tab's tooltip
28943      */
28944     setTooltip : function(text){
28945         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28946             this.textEl.dom.qtip = text;
28947             this.textEl.dom.removeAttribute('title');
28948         }else{
28949             this.textEl.dom.title = text;
28950         }
28951     },
28952
28953     onTabClick : function(e){
28954         e.preventDefault();
28955         this.tabPanel.activate(this.id);
28956     },
28957
28958     onTabMouseDown : function(e){
28959         e.preventDefault();
28960         this.tabPanel.activate(this.id);
28961     },
28962
28963     getWidth : function(){
28964         return this.inner.getWidth();
28965     },
28966
28967     setWidth : function(width){
28968         var iwidth = width - this.pnode.getPadding("lr");
28969         this.inner.setWidth(iwidth);
28970         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28971         this.pnode.setWidth(width);
28972     },
28973
28974     /**
28975      * Show or hide the tab
28976      * @param {Boolean} hidden True to hide or false to show.
28977      */
28978     setHidden : function(hidden){
28979         this.hidden = hidden;
28980         this.pnode.setStyle("display", hidden ? "none" : "");
28981     },
28982
28983     /**
28984      * Returns true if this tab is "hidden"
28985      * @return {Boolean}
28986      */
28987     isHidden : function(){
28988         return this.hidden;
28989     },
28990
28991     /**
28992      * Returns the text for this tab
28993      * @return {String}
28994      */
28995     getText : function(){
28996         return this.text;
28997     },
28998
28999     autoSize : function(){
29000         //this.el.beginMeasure();
29001         this.textEl.setWidth(1);
29002         /*
29003          *  #2804 [new] Tabs in Roojs
29004          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
29005          */
29006         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
29007         //this.el.endMeasure();
29008     },
29009
29010     /**
29011      * Sets the text for the tab (Note: this also sets the tooltip text)
29012      * @param {String} text The tab's text and tooltip
29013      */
29014     setText : function(text){
29015         this.text = text;
29016         this.textEl.update(text);
29017         this.setTooltip(text);
29018         if(!this.tabPanel.resizeTabs){
29019             this.autoSize();
29020         }
29021     },
29022     /**
29023      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
29024      */
29025     activate : function(){
29026         this.tabPanel.activate(this.id);
29027     },
29028
29029     /**
29030      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
29031      */
29032     disable : function(){
29033         if(this.tabPanel.active != this){
29034             this.disabled = true;
29035             this.pnode.addClass("disabled");
29036         }
29037     },
29038
29039     /**
29040      * Enables this TabPanelItem if it was previously disabled.
29041      */
29042     enable : function(){
29043         this.disabled = false;
29044         this.pnode.removeClass("disabled");
29045     },
29046
29047     /**
29048      * Sets the content for this TabPanelItem.
29049      * @param {String} content The content
29050      * @param {Boolean} loadScripts true to look for and load scripts
29051      */
29052     setContent : function(content, loadScripts){
29053         this.bodyEl.update(content, loadScripts);
29054     },
29055
29056     /**
29057      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29058      * @return {Roo.UpdateManager} The UpdateManager
29059      */
29060     getUpdateManager : function(){
29061         return this.bodyEl.getUpdateManager();
29062     },
29063
29064     /**
29065      * Set a URL to be used to load the content for this TabPanelItem.
29066      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29067      * @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)
29068      * @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)
29069      * @return {Roo.UpdateManager} The UpdateManager
29070      */
29071     setUrl : function(url, params, loadOnce){
29072         if(this.refreshDelegate){
29073             this.un('activate', this.refreshDelegate);
29074         }
29075         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29076         this.on("activate", this.refreshDelegate);
29077         return this.bodyEl.getUpdateManager();
29078     },
29079
29080     /** @private */
29081     _handleRefresh : function(url, params, loadOnce){
29082         if(!loadOnce || !this.loaded){
29083             var updater = this.bodyEl.getUpdateManager();
29084             updater.update(url, params, this._setLoaded.createDelegate(this));
29085         }
29086     },
29087
29088     /**
29089      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29090      *   Will fail silently if the setUrl method has not been called.
29091      *   This does not activate the panel, just updates its content.
29092      */
29093     refresh : function(){
29094         if(this.refreshDelegate){
29095            this.loaded = false;
29096            this.refreshDelegate();
29097         }
29098     },
29099
29100     /** @private */
29101     _setLoaded : function(){
29102         this.loaded = true;
29103     },
29104
29105     /** @private */
29106     closeClick : function(e){
29107         var o = {};
29108         e.stopEvent();
29109         this.fireEvent("beforeclose", this, o);
29110         if(o.cancel !== true){
29111             this.tabPanel.removeTab(this.id);
29112         }
29113     },
29114     /**
29115      * The text displayed in the tooltip for the close icon.
29116      * @type String
29117      */
29118     closeText : "Close this tab"
29119 });
29120
29121 /** @private */
29122 Roo.TabPanel.prototype.createStrip = function(container){
29123     var strip = document.createElement("div");
29124     strip.className = "x-tabs-wrap";
29125     container.appendChild(strip);
29126     return strip;
29127 };
29128 /** @private */
29129 Roo.TabPanel.prototype.createStripList = function(strip){
29130     // div wrapper for retard IE
29131     // returns the "tr" element.
29132     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29133         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29134         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29135     return strip.firstChild.firstChild.firstChild.firstChild;
29136 };
29137 /** @private */
29138 Roo.TabPanel.prototype.createBody = function(container){
29139     var body = document.createElement("div");
29140     Roo.id(body, "tab-body");
29141     Roo.fly(body).addClass("x-tabs-body");
29142     container.appendChild(body);
29143     return body;
29144 };
29145 /** @private */
29146 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29147     var body = Roo.getDom(id);
29148     if(!body){
29149         body = document.createElement("div");
29150         body.id = id;
29151     }
29152     Roo.fly(body).addClass("x-tabs-item-body");
29153     bodyEl.insertBefore(body, bodyEl.firstChild);
29154     return body;
29155 };
29156 /** @private */
29157 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29158     var td = document.createElement("td");
29159     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29160     //stripEl.appendChild(td);
29161     if(closable){
29162         td.className = "x-tabs-closable";
29163         if(!this.closeTpl){
29164             this.closeTpl = new Roo.Template(
29165                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29166                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29167                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29168             );
29169         }
29170         var el = this.closeTpl.overwrite(td, {"text": text});
29171         var close = el.getElementsByTagName("div")[0];
29172         var inner = el.getElementsByTagName("em")[0];
29173         return {"el": el, "close": close, "inner": inner};
29174     } else {
29175         if(!this.tabTpl){
29176             this.tabTpl = new Roo.Template(
29177                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29178                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29179             );
29180         }
29181         var el = this.tabTpl.overwrite(td, {"text": text});
29182         var inner = el.getElementsByTagName("em")[0];
29183         return {"el": el, "inner": inner};
29184     }
29185 };/*
29186  * Based on:
29187  * Ext JS Library 1.1.1
29188  * Copyright(c) 2006-2007, Ext JS, LLC.
29189  *
29190  * Originally Released Under LGPL - original licence link has changed is not relivant.
29191  *
29192  * Fork - LGPL
29193  * <script type="text/javascript">
29194  */
29195
29196 /**
29197  * @class Roo.Button
29198  * @extends Roo.util.Observable
29199  * Simple Button class
29200  * @cfg {String} text The button text
29201  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29202  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29203  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29204  * @cfg {Object} scope The scope of the handler
29205  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29206  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29207  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29208  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29209  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29210  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29211    applies if enableToggle = true)
29212  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29213  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29214   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29215  * @constructor
29216  * Create a new button
29217  * @param {Object} config The config object
29218  */
29219 Roo.Button = function(renderTo, config)
29220 {
29221     if (!config) {
29222         config = renderTo;
29223         renderTo = config.renderTo || false;
29224     }
29225     
29226     Roo.apply(this, config);
29227     this.addEvents({
29228         /**
29229              * @event click
29230              * Fires when this button is clicked
29231              * @param {Button} this
29232              * @param {EventObject} e The click event
29233              */
29234             "click" : true,
29235         /**
29236              * @event toggle
29237              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29238              * @param {Button} this
29239              * @param {Boolean} pressed
29240              */
29241             "toggle" : true,
29242         /**
29243              * @event mouseover
29244              * Fires when the mouse hovers over the button
29245              * @param {Button} this
29246              * @param {Event} e The event object
29247              */
29248         'mouseover' : true,
29249         /**
29250              * @event mouseout
29251              * Fires when the mouse exits the button
29252              * @param {Button} this
29253              * @param {Event} e The event object
29254              */
29255         'mouseout': true,
29256          /**
29257              * @event render
29258              * Fires when the button is rendered
29259              * @param {Button} this
29260              */
29261         'render': true
29262     });
29263     if(this.menu){
29264         this.menu = Roo.menu.MenuMgr.get(this.menu);
29265     }
29266     // register listeners first!!  - so render can be captured..
29267     Roo.util.Observable.call(this);
29268     if(renderTo){
29269         this.render(renderTo);
29270     }
29271     
29272   
29273 };
29274
29275 Roo.extend(Roo.Button, Roo.util.Observable, {
29276     /**
29277      * 
29278      */
29279     
29280     /**
29281      * Read-only. True if this button is hidden
29282      * @type Boolean
29283      */
29284     hidden : false,
29285     /**
29286      * Read-only. True if this button is disabled
29287      * @type Boolean
29288      */
29289     disabled : false,
29290     /**
29291      * Read-only. True if this button is pressed (only if enableToggle = true)
29292      * @type Boolean
29293      */
29294     pressed : false,
29295
29296     /**
29297      * @cfg {Number} tabIndex 
29298      * The DOM tabIndex for this button (defaults to undefined)
29299      */
29300     tabIndex : undefined,
29301
29302     /**
29303      * @cfg {Boolean} enableToggle
29304      * True to enable pressed/not pressed toggling (defaults to false)
29305      */
29306     enableToggle: false,
29307     /**
29308      * @cfg {Mixed} menu
29309      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29310      */
29311     menu : undefined,
29312     /**
29313      * @cfg {String} menuAlign
29314      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29315      */
29316     menuAlign : "tl-bl?",
29317
29318     /**
29319      * @cfg {String} iconCls
29320      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29321      */
29322     iconCls : undefined,
29323     /**
29324      * @cfg {String} type
29325      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29326      */
29327     type : 'button',
29328
29329     // private
29330     menuClassTarget: 'tr',
29331
29332     /**
29333      * @cfg {String} clickEvent
29334      * The type of event to map to the button's event handler (defaults to 'click')
29335      */
29336     clickEvent : 'click',
29337
29338     /**
29339      * @cfg {Boolean} handleMouseEvents
29340      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29341      */
29342     handleMouseEvents : true,
29343
29344     /**
29345      * @cfg {String} tooltipType
29346      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29347      */
29348     tooltipType : 'qtip',
29349
29350     /**
29351      * @cfg {String} cls
29352      * A CSS class to apply to the button's main element.
29353      */
29354     
29355     /**
29356      * @cfg {Roo.Template} template (Optional)
29357      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29358      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29359      * require code modifications if required elements (e.g. a button) aren't present.
29360      */
29361
29362     // private
29363     render : function(renderTo){
29364         var btn;
29365         if(this.hideParent){
29366             this.parentEl = Roo.get(renderTo);
29367         }
29368         if(!this.dhconfig){
29369             if(!this.template){
29370                 if(!Roo.Button.buttonTemplate){
29371                     // hideous table template
29372                     Roo.Button.buttonTemplate = new Roo.Template(
29373                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29374                         '<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>',
29375                         "</tr></tbody></table>");
29376                 }
29377                 this.template = Roo.Button.buttonTemplate;
29378             }
29379             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29380             var btnEl = btn.child("button:first");
29381             btnEl.on('focus', this.onFocus, this);
29382             btnEl.on('blur', this.onBlur, this);
29383             if(this.cls){
29384                 btn.addClass(this.cls);
29385             }
29386             if(this.icon){
29387                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29388             }
29389             if(this.iconCls){
29390                 btnEl.addClass(this.iconCls);
29391                 if(!this.cls){
29392                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29393                 }
29394             }
29395             if(this.tabIndex !== undefined){
29396                 btnEl.dom.tabIndex = this.tabIndex;
29397             }
29398             if(this.tooltip){
29399                 if(typeof this.tooltip == 'object'){
29400                     Roo.QuickTips.tips(Roo.apply({
29401                           target: btnEl.id
29402                     }, this.tooltip));
29403                 } else {
29404                     btnEl.dom[this.tooltipType] = this.tooltip;
29405                 }
29406             }
29407         }else{
29408             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29409         }
29410         this.el = btn;
29411         if(this.id){
29412             this.el.dom.id = this.el.id = this.id;
29413         }
29414         if(this.menu){
29415             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29416             this.menu.on("show", this.onMenuShow, this);
29417             this.menu.on("hide", this.onMenuHide, this);
29418         }
29419         btn.addClass("x-btn");
29420         if(Roo.isIE && !Roo.isIE7){
29421             this.autoWidth.defer(1, this);
29422         }else{
29423             this.autoWidth();
29424         }
29425         if(this.handleMouseEvents){
29426             btn.on("mouseover", this.onMouseOver, this);
29427             btn.on("mouseout", this.onMouseOut, this);
29428             btn.on("mousedown", this.onMouseDown, this);
29429         }
29430         btn.on(this.clickEvent, this.onClick, this);
29431         //btn.on("mouseup", this.onMouseUp, this);
29432         if(this.hidden){
29433             this.hide();
29434         }
29435         if(this.disabled){
29436             this.disable();
29437         }
29438         Roo.ButtonToggleMgr.register(this);
29439         if(this.pressed){
29440             this.el.addClass("x-btn-pressed");
29441         }
29442         if(this.repeat){
29443             var repeater = new Roo.util.ClickRepeater(btn,
29444                 typeof this.repeat == "object" ? this.repeat : {}
29445             );
29446             repeater.on("click", this.onClick,  this);
29447         }
29448         
29449         this.fireEvent('render', this);
29450         
29451     },
29452     /**
29453      * Returns the button's underlying element
29454      * @return {Roo.Element} The element
29455      */
29456     getEl : function(){
29457         return this.el;  
29458     },
29459     
29460     /**
29461      * Destroys this Button and removes any listeners.
29462      */
29463     destroy : function(){
29464         Roo.ButtonToggleMgr.unregister(this);
29465         this.el.removeAllListeners();
29466         this.purgeListeners();
29467         this.el.remove();
29468     },
29469
29470     // private
29471     autoWidth : function(){
29472         if(this.el){
29473             this.el.setWidth("auto");
29474             if(Roo.isIE7 && Roo.isStrict){
29475                 var ib = this.el.child('button');
29476                 if(ib && ib.getWidth() > 20){
29477                     ib.clip();
29478                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29479                 }
29480             }
29481             if(this.minWidth){
29482                 if(this.hidden){
29483                     this.el.beginMeasure();
29484                 }
29485                 if(this.el.getWidth() < this.minWidth){
29486                     this.el.setWidth(this.minWidth);
29487                 }
29488                 if(this.hidden){
29489                     this.el.endMeasure();
29490                 }
29491             }
29492         }
29493     },
29494
29495     /**
29496      * Assigns this button's click handler
29497      * @param {Function} handler The function to call when the button is clicked
29498      * @param {Object} scope (optional) Scope for the function passed in
29499      */
29500     setHandler : function(handler, scope){
29501         this.handler = handler;
29502         this.scope = scope;  
29503     },
29504     
29505     /**
29506      * Sets this button's text
29507      * @param {String} text The button text
29508      */
29509     setText : function(text){
29510         this.text = text;
29511         if(this.el){
29512             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29513         }
29514         this.autoWidth();
29515     },
29516     
29517     /**
29518      * Gets the text for this button
29519      * @return {String} The button text
29520      */
29521     getText : function(){
29522         return this.text;  
29523     },
29524     
29525     /**
29526      * Show this button
29527      */
29528     show: function(){
29529         this.hidden = false;
29530         if(this.el){
29531             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29532         }
29533     },
29534     
29535     /**
29536      * Hide this button
29537      */
29538     hide: function(){
29539         this.hidden = true;
29540         if(this.el){
29541             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29542         }
29543     },
29544     
29545     /**
29546      * Convenience function for boolean show/hide
29547      * @param {Boolean} visible True to show, false to hide
29548      */
29549     setVisible: function(visible){
29550         if(visible) {
29551             this.show();
29552         }else{
29553             this.hide();
29554         }
29555     },
29556     
29557     /**
29558      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29559      * @param {Boolean} state (optional) Force a particular state
29560      */
29561     toggle : function(state){
29562         state = state === undefined ? !this.pressed : state;
29563         if(state != this.pressed){
29564             if(state){
29565                 this.el.addClass("x-btn-pressed");
29566                 this.pressed = true;
29567                 this.fireEvent("toggle", this, true);
29568             }else{
29569                 this.el.removeClass("x-btn-pressed");
29570                 this.pressed = false;
29571                 this.fireEvent("toggle", this, false);
29572             }
29573             if(this.toggleHandler){
29574                 this.toggleHandler.call(this.scope || this, this, state);
29575             }
29576         }
29577     },
29578     
29579     /**
29580      * Focus the button
29581      */
29582     focus : function(){
29583         this.el.child('button:first').focus();
29584     },
29585     
29586     /**
29587      * Disable this button
29588      */
29589     disable : function(){
29590         if(this.el){
29591             this.el.addClass("x-btn-disabled");
29592         }
29593         this.disabled = true;
29594     },
29595     
29596     /**
29597      * Enable this button
29598      */
29599     enable : function(){
29600         if(this.el){
29601             this.el.removeClass("x-btn-disabled");
29602         }
29603         this.disabled = false;
29604     },
29605
29606     /**
29607      * Convenience function for boolean enable/disable
29608      * @param {Boolean} enabled True to enable, false to disable
29609      */
29610     setDisabled : function(v){
29611         this[v !== true ? "enable" : "disable"]();
29612     },
29613
29614     // private
29615     onClick : function(e)
29616     {
29617         if(e){
29618             e.preventDefault();
29619         }
29620         if(e.button != 0){
29621             return;
29622         }
29623         if(!this.disabled){
29624             if(this.enableToggle){
29625                 this.toggle();
29626             }
29627             if(this.menu && !this.menu.isVisible()){
29628                 this.menu.show(this.el, this.menuAlign);
29629             }
29630             this.fireEvent("click", this, e);
29631             if(this.handler){
29632                 this.el.removeClass("x-btn-over");
29633                 this.handler.call(this.scope || this, this, e);
29634             }
29635         }
29636     },
29637     // private
29638     onMouseOver : function(e){
29639         if(!this.disabled){
29640             this.el.addClass("x-btn-over");
29641             this.fireEvent('mouseover', this, e);
29642         }
29643     },
29644     // private
29645     onMouseOut : function(e){
29646         if(!e.within(this.el,  true)){
29647             this.el.removeClass("x-btn-over");
29648             this.fireEvent('mouseout', this, e);
29649         }
29650     },
29651     // private
29652     onFocus : function(e){
29653         if(!this.disabled){
29654             this.el.addClass("x-btn-focus");
29655         }
29656     },
29657     // private
29658     onBlur : function(e){
29659         this.el.removeClass("x-btn-focus");
29660     },
29661     // private
29662     onMouseDown : function(e){
29663         if(!this.disabled && e.button == 0){
29664             this.el.addClass("x-btn-click");
29665             Roo.get(document).on('mouseup', this.onMouseUp, this);
29666         }
29667     },
29668     // private
29669     onMouseUp : function(e){
29670         if(e.button == 0){
29671             this.el.removeClass("x-btn-click");
29672             Roo.get(document).un('mouseup', this.onMouseUp, this);
29673         }
29674     },
29675     // private
29676     onMenuShow : function(e){
29677         this.el.addClass("x-btn-menu-active");
29678     },
29679     // private
29680     onMenuHide : function(e){
29681         this.el.removeClass("x-btn-menu-active");
29682     }   
29683 });
29684
29685 // Private utility class used by Button
29686 Roo.ButtonToggleMgr = function(){
29687    var groups = {};
29688    
29689    function toggleGroup(btn, state){
29690        if(state){
29691            var g = groups[btn.toggleGroup];
29692            for(var i = 0, l = g.length; i < l; i++){
29693                if(g[i] != btn){
29694                    g[i].toggle(false);
29695                }
29696            }
29697        }
29698    }
29699    
29700    return {
29701        register : function(btn){
29702            if(!btn.toggleGroup){
29703                return;
29704            }
29705            var g = groups[btn.toggleGroup];
29706            if(!g){
29707                g = groups[btn.toggleGroup] = [];
29708            }
29709            g.push(btn);
29710            btn.on("toggle", toggleGroup);
29711        },
29712        
29713        unregister : function(btn){
29714            if(!btn.toggleGroup){
29715                return;
29716            }
29717            var g = groups[btn.toggleGroup];
29718            if(g){
29719                g.remove(btn);
29720                btn.un("toggle", toggleGroup);
29721            }
29722        }
29723    };
29724 }();/*
29725  * Based on:
29726  * Ext JS Library 1.1.1
29727  * Copyright(c) 2006-2007, Ext JS, LLC.
29728  *
29729  * Originally Released Under LGPL - original licence link has changed is not relivant.
29730  *
29731  * Fork - LGPL
29732  * <script type="text/javascript">
29733  */
29734  
29735 /**
29736  * @class Roo.SplitButton
29737  * @extends Roo.Button
29738  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29739  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29740  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29741  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29742  * @cfg {String} arrowTooltip The title attribute of the arrow
29743  * @constructor
29744  * Create a new menu button
29745  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29746  * @param {Object} config The config object
29747  */
29748 Roo.SplitButton = function(renderTo, config){
29749     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29750     /**
29751      * @event arrowclick
29752      * Fires when this button's arrow is clicked
29753      * @param {SplitButton} this
29754      * @param {EventObject} e The click event
29755      */
29756     this.addEvents({"arrowclick":true});
29757 };
29758
29759 Roo.extend(Roo.SplitButton, Roo.Button, {
29760     render : function(renderTo){
29761         // this is one sweet looking template!
29762         var tpl = new Roo.Template(
29763             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29764             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29765             '<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>',
29766             "</tbody></table></td><td>",
29767             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29768             '<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>',
29769             "</tbody></table></td></tr></table>"
29770         );
29771         var btn = tpl.append(renderTo, [this.text, this.type], true);
29772         var btnEl = btn.child("button");
29773         if(this.cls){
29774             btn.addClass(this.cls);
29775         }
29776         if(this.icon){
29777             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29778         }
29779         if(this.iconCls){
29780             btnEl.addClass(this.iconCls);
29781             if(!this.cls){
29782                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29783             }
29784         }
29785         this.el = btn;
29786         if(this.handleMouseEvents){
29787             btn.on("mouseover", this.onMouseOver, this);
29788             btn.on("mouseout", this.onMouseOut, this);
29789             btn.on("mousedown", this.onMouseDown, this);
29790             btn.on("mouseup", this.onMouseUp, this);
29791         }
29792         btn.on(this.clickEvent, this.onClick, this);
29793         if(this.tooltip){
29794             if(typeof this.tooltip == 'object'){
29795                 Roo.QuickTips.tips(Roo.apply({
29796                       target: btnEl.id
29797                 }, this.tooltip));
29798             } else {
29799                 btnEl.dom[this.tooltipType] = this.tooltip;
29800             }
29801         }
29802         if(this.arrowTooltip){
29803             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29804         }
29805         if(this.hidden){
29806             this.hide();
29807         }
29808         if(this.disabled){
29809             this.disable();
29810         }
29811         if(this.pressed){
29812             this.el.addClass("x-btn-pressed");
29813         }
29814         if(Roo.isIE && !Roo.isIE7){
29815             this.autoWidth.defer(1, this);
29816         }else{
29817             this.autoWidth();
29818         }
29819         if(this.menu){
29820             this.menu.on("show", this.onMenuShow, this);
29821             this.menu.on("hide", this.onMenuHide, this);
29822         }
29823         this.fireEvent('render', this);
29824     },
29825
29826     // private
29827     autoWidth : function(){
29828         if(this.el){
29829             var tbl = this.el.child("table:first");
29830             var tbl2 = this.el.child("table:last");
29831             this.el.setWidth("auto");
29832             tbl.setWidth("auto");
29833             if(Roo.isIE7 && Roo.isStrict){
29834                 var ib = this.el.child('button:first');
29835                 if(ib && ib.getWidth() > 20){
29836                     ib.clip();
29837                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29838                 }
29839             }
29840             if(this.minWidth){
29841                 if(this.hidden){
29842                     this.el.beginMeasure();
29843                 }
29844                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29845                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29846                 }
29847                 if(this.hidden){
29848                     this.el.endMeasure();
29849                 }
29850             }
29851             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29852         } 
29853     },
29854     /**
29855      * Sets this button's click handler
29856      * @param {Function} handler The function to call when the button is clicked
29857      * @param {Object} scope (optional) Scope for the function passed above
29858      */
29859     setHandler : function(handler, scope){
29860         this.handler = handler;
29861         this.scope = scope;  
29862     },
29863     
29864     /**
29865      * Sets this button's arrow click handler
29866      * @param {Function} handler The function to call when the arrow is clicked
29867      * @param {Object} scope (optional) Scope for the function passed above
29868      */
29869     setArrowHandler : function(handler, scope){
29870         this.arrowHandler = handler;
29871         this.scope = scope;  
29872     },
29873     
29874     /**
29875      * Focus the button
29876      */
29877     focus : function(){
29878         if(this.el){
29879             this.el.child("button:first").focus();
29880         }
29881     },
29882
29883     // private
29884     onClick : function(e){
29885         e.preventDefault();
29886         if(!this.disabled){
29887             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29888                 if(this.menu && !this.menu.isVisible()){
29889                     this.menu.show(this.el, this.menuAlign);
29890                 }
29891                 this.fireEvent("arrowclick", this, e);
29892                 if(this.arrowHandler){
29893                     this.arrowHandler.call(this.scope || this, this, e);
29894                 }
29895             }else{
29896                 this.fireEvent("click", this, e);
29897                 if(this.handler){
29898                     this.handler.call(this.scope || this, this, e);
29899                 }
29900             }
29901         }
29902     },
29903     // private
29904     onMouseDown : function(e){
29905         if(!this.disabled){
29906             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29907         }
29908     },
29909     // private
29910     onMouseUp : function(e){
29911         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29912     }   
29913 });
29914
29915
29916 // backwards compat
29917 Roo.MenuButton = Roo.SplitButton;/*
29918  * Based on:
29919  * Ext JS Library 1.1.1
29920  * Copyright(c) 2006-2007, Ext JS, LLC.
29921  *
29922  * Originally Released Under LGPL - original licence link has changed is not relivant.
29923  *
29924  * Fork - LGPL
29925  * <script type="text/javascript">
29926  */
29927
29928 /**
29929  * @class Roo.Toolbar
29930  * Basic Toolbar class.
29931  * @constructor
29932  * Creates a new Toolbar
29933  * @param {Object} container The config object
29934  */ 
29935 Roo.Toolbar = function(container, buttons, config)
29936 {
29937     /// old consturctor format still supported..
29938     if(container instanceof Array){ // omit the container for later rendering
29939         buttons = container;
29940         config = buttons;
29941         container = null;
29942     }
29943     if (typeof(container) == 'object' && container.xtype) {
29944         config = container;
29945         container = config.container;
29946         buttons = config.buttons || []; // not really - use items!!
29947     }
29948     var xitems = [];
29949     if (config && config.items) {
29950         xitems = config.items;
29951         delete config.items;
29952     }
29953     Roo.apply(this, config);
29954     this.buttons = buttons;
29955     
29956     if(container){
29957         this.render(container);
29958     }
29959     this.xitems = xitems;
29960     Roo.each(xitems, function(b) {
29961         this.add(b);
29962     }, this);
29963     
29964 };
29965
29966 Roo.Toolbar.prototype = {
29967     /**
29968      * @cfg {Array} items
29969      * array of button configs or elements to add (will be converted to a MixedCollection)
29970      */
29971     
29972     /**
29973      * @cfg {String/HTMLElement/Element} container
29974      * The id or element that will contain the toolbar
29975      */
29976     // private
29977     render : function(ct){
29978         this.el = Roo.get(ct);
29979         if(this.cls){
29980             this.el.addClass(this.cls);
29981         }
29982         // using a table allows for vertical alignment
29983         // 100% width is needed by Safari...
29984         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29985         this.tr = this.el.child("tr", true);
29986         var autoId = 0;
29987         this.items = new Roo.util.MixedCollection(false, function(o){
29988             return o.id || ("item" + (++autoId));
29989         });
29990         if(this.buttons){
29991             this.add.apply(this, this.buttons);
29992             delete this.buttons;
29993         }
29994     },
29995
29996     /**
29997      * Adds element(s) to the toolbar -- this function takes a variable number of 
29998      * arguments of mixed type and adds them to the toolbar.
29999      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
30000      * <ul>
30001      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
30002      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
30003      * <li>Field: Any form field (equivalent to {@link #addField})</li>
30004      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
30005      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
30006      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
30007      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
30008      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
30009      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
30010      * </ul>
30011      * @param {Mixed} arg2
30012      * @param {Mixed} etc.
30013      */
30014     add : function(){
30015         var a = arguments, l = a.length;
30016         for(var i = 0; i < l; i++){
30017             this._add(a[i]);
30018         }
30019     },
30020     // private..
30021     _add : function(el) {
30022         
30023         if (el.xtype) {
30024             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
30025         }
30026         
30027         if (el.applyTo){ // some kind of form field
30028             return this.addField(el);
30029         } 
30030         if (el.render){ // some kind of Toolbar.Item
30031             return this.addItem(el);
30032         }
30033         if (typeof el == "string"){ // string
30034             if(el == "separator" || el == "-"){
30035                 return this.addSeparator();
30036             }
30037             if (el == " "){
30038                 return this.addSpacer();
30039             }
30040             if(el == "->"){
30041                 return this.addFill();
30042             }
30043             return this.addText(el);
30044             
30045         }
30046         if(el.tagName){ // element
30047             return this.addElement(el);
30048         }
30049         if(typeof el == "object"){ // must be button config?
30050             return this.addButton(el);
30051         }
30052         // and now what?!?!
30053         return false;
30054         
30055     },
30056     
30057     /**
30058      * Add an Xtype element
30059      * @param {Object} xtype Xtype Object
30060      * @return {Object} created Object
30061      */
30062     addxtype : function(e){
30063         return this.add(e);  
30064     },
30065     
30066     /**
30067      * Returns the Element for this toolbar.
30068      * @return {Roo.Element}
30069      */
30070     getEl : function(){
30071         return this.el;  
30072     },
30073     
30074     /**
30075      * Adds a separator
30076      * @return {Roo.Toolbar.Item} The separator item
30077      */
30078     addSeparator : function(){
30079         return this.addItem(new Roo.Toolbar.Separator());
30080     },
30081
30082     /**
30083      * Adds a spacer element
30084      * @return {Roo.Toolbar.Spacer} The spacer item
30085      */
30086     addSpacer : function(){
30087         return this.addItem(new Roo.Toolbar.Spacer());
30088     },
30089
30090     /**
30091      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30092      * @return {Roo.Toolbar.Fill} The fill item
30093      */
30094     addFill : function(){
30095         return this.addItem(new Roo.Toolbar.Fill());
30096     },
30097
30098     /**
30099      * Adds any standard HTML element to the toolbar
30100      * @param {String/HTMLElement/Element} el The element or id of the element to add
30101      * @return {Roo.Toolbar.Item} The element's item
30102      */
30103     addElement : function(el){
30104         return this.addItem(new Roo.Toolbar.Item(el));
30105     },
30106     /**
30107      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30108      * @type Roo.util.MixedCollection  
30109      */
30110     items : false,
30111      
30112     /**
30113      * Adds any Toolbar.Item or subclass
30114      * @param {Roo.Toolbar.Item} item
30115      * @return {Roo.Toolbar.Item} The item
30116      */
30117     addItem : function(item){
30118         var td = this.nextBlock();
30119         item.render(td);
30120         this.items.add(item);
30121         return item;
30122     },
30123     
30124     /**
30125      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30126      * @param {Object/Array} config A button config or array of configs
30127      * @return {Roo.Toolbar.Button/Array}
30128      */
30129     addButton : function(config){
30130         if(config instanceof Array){
30131             var buttons = [];
30132             for(var i = 0, len = config.length; i < len; i++) {
30133                 buttons.push(this.addButton(config[i]));
30134             }
30135             return buttons;
30136         }
30137         var b = config;
30138         if(!(config instanceof Roo.Toolbar.Button)){
30139             b = config.split ?
30140                 new Roo.Toolbar.SplitButton(config) :
30141                 new Roo.Toolbar.Button(config);
30142         }
30143         var td = this.nextBlock();
30144         b.render(td);
30145         this.items.add(b);
30146         return b;
30147     },
30148     
30149     /**
30150      * Adds text to the toolbar
30151      * @param {String} text The text to add
30152      * @return {Roo.Toolbar.Item} The element's item
30153      */
30154     addText : function(text){
30155         return this.addItem(new Roo.Toolbar.TextItem(text));
30156     },
30157     
30158     /**
30159      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30160      * @param {Number} index The index where the item is to be inserted
30161      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30162      * @return {Roo.Toolbar.Button/Item}
30163      */
30164     insertButton : function(index, item){
30165         if(item instanceof Array){
30166             var buttons = [];
30167             for(var i = 0, len = item.length; i < len; i++) {
30168                buttons.push(this.insertButton(index + i, item[i]));
30169             }
30170             return buttons;
30171         }
30172         if (!(item instanceof Roo.Toolbar.Button)){
30173            item = new Roo.Toolbar.Button(item);
30174         }
30175         var td = document.createElement("td");
30176         this.tr.insertBefore(td, this.tr.childNodes[index]);
30177         item.render(td);
30178         this.items.insert(index, item);
30179         return item;
30180     },
30181     
30182     /**
30183      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30184      * @param {Object} config
30185      * @return {Roo.Toolbar.Item} The element's item
30186      */
30187     addDom : function(config, returnEl){
30188         var td = this.nextBlock();
30189         Roo.DomHelper.overwrite(td, config);
30190         var ti = new Roo.Toolbar.Item(td.firstChild);
30191         ti.render(td);
30192         this.items.add(ti);
30193         return ti;
30194     },
30195
30196     /**
30197      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30198      * @type Roo.util.MixedCollection  
30199      */
30200     fields : false,
30201     
30202     /**
30203      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30204      * Note: the field should not have been rendered yet. For a field that has already been
30205      * rendered, use {@link #addElement}.
30206      * @param {Roo.form.Field} field
30207      * @return {Roo.ToolbarItem}
30208      */
30209      
30210       
30211     addField : function(field) {
30212         if (!this.fields) {
30213             var autoId = 0;
30214             this.fields = new Roo.util.MixedCollection(false, function(o){
30215                 return o.id || ("item" + (++autoId));
30216             });
30217
30218         }
30219         
30220         var td = this.nextBlock();
30221         field.render(td);
30222         var ti = new Roo.Toolbar.Item(td.firstChild);
30223         ti.render(td);
30224         this.items.add(ti);
30225         this.fields.add(field);
30226         return ti;
30227     },
30228     /**
30229      * Hide the toolbar
30230      * @method hide
30231      */
30232      
30233       
30234     hide : function()
30235     {
30236         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30237         this.el.child('div').hide();
30238     },
30239     /**
30240      * Show the toolbar
30241      * @method show
30242      */
30243     show : function()
30244     {
30245         this.el.child('div').show();
30246     },
30247       
30248     // private
30249     nextBlock : function(){
30250         var td = document.createElement("td");
30251         this.tr.appendChild(td);
30252         return td;
30253     },
30254
30255     // private
30256     destroy : function(){
30257         if(this.items){ // rendered?
30258             Roo.destroy.apply(Roo, this.items.items);
30259         }
30260         if(this.fields){ // rendered?
30261             Roo.destroy.apply(Roo, this.fields.items);
30262         }
30263         Roo.Element.uncache(this.el, this.tr);
30264     }
30265 };
30266
30267 /**
30268  * @class Roo.Toolbar.Item
30269  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30270  * @constructor
30271  * Creates a new Item
30272  * @param {HTMLElement} el 
30273  */
30274 Roo.Toolbar.Item = function(el){
30275     var cfg = {};
30276     if (typeof (el.xtype) != 'undefined') {
30277         cfg = el;
30278         el = cfg.el;
30279     }
30280     
30281     this.el = Roo.getDom(el);
30282     this.id = Roo.id(this.el);
30283     this.hidden = false;
30284     
30285     this.addEvents({
30286          /**
30287              * @event render
30288              * Fires when the button is rendered
30289              * @param {Button} this
30290              */
30291         'render': true
30292     });
30293     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30294 };
30295 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30296 //Roo.Toolbar.Item.prototype = {
30297     
30298     /**
30299      * Get this item's HTML Element
30300      * @return {HTMLElement}
30301      */
30302     getEl : function(){
30303        return this.el;  
30304     },
30305
30306     // private
30307     render : function(td){
30308         
30309          this.td = td;
30310         td.appendChild(this.el);
30311         
30312         this.fireEvent('render', this);
30313     },
30314     
30315     /**
30316      * Removes and destroys this item.
30317      */
30318     destroy : function(){
30319         this.td.parentNode.removeChild(this.td);
30320     },
30321     
30322     /**
30323      * Shows this item.
30324      */
30325     show: function(){
30326         this.hidden = false;
30327         this.td.style.display = "";
30328     },
30329     
30330     /**
30331      * Hides this item.
30332      */
30333     hide: function(){
30334         this.hidden = true;
30335         this.td.style.display = "none";
30336     },
30337     
30338     /**
30339      * Convenience function for boolean show/hide.
30340      * @param {Boolean} visible true to show/false to hide
30341      */
30342     setVisible: function(visible){
30343         if(visible) {
30344             this.show();
30345         }else{
30346             this.hide();
30347         }
30348     },
30349     
30350     /**
30351      * Try to focus this item.
30352      */
30353     focus : function(){
30354         Roo.fly(this.el).focus();
30355     },
30356     
30357     /**
30358      * Disables this item.
30359      */
30360     disable : function(){
30361         Roo.fly(this.td).addClass("x-item-disabled");
30362         this.disabled = true;
30363         this.el.disabled = true;
30364     },
30365     
30366     /**
30367      * Enables this item.
30368      */
30369     enable : function(){
30370         Roo.fly(this.td).removeClass("x-item-disabled");
30371         this.disabled = false;
30372         this.el.disabled = false;
30373     }
30374 });
30375
30376
30377 /**
30378  * @class Roo.Toolbar.Separator
30379  * @extends Roo.Toolbar.Item
30380  * A simple toolbar separator class
30381  * @constructor
30382  * Creates a new Separator
30383  */
30384 Roo.Toolbar.Separator = function(cfg){
30385     
30386     var s = document.createElement("span");
30387     s.className = "ytb-sep";
30388     if (cfg) {
30389         cfg.el = s;
30390     }
30391     
30392     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30393 };
30394 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30395     enable:Roo.emptyFn,
30396     disable:Roo.emptyFn,
30397     focus:Roo.emptyFn
30398 });
30399
30400 /**
30401  * @class Roo.Toolbar.Spacer
30402  * @extends Roo.Toolbar.Item
30403  * A simple element that adds extra horizontal space to a toolbar.
30404  * @constructor
30405  * Creates a new Spacer
30406  */
30407 Roo.Toolbar.Spacer = function(cfg){
30408     var s = document.createElement("div");
30409     s.className = "ytb-spacer";
30410     if (cfg) {
30411         cfg.el = s;
30412     }
30413     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30414 };
30415 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30416     enable:Roo.emptyFn,
30417     disable:Roo.emptyFn,
30418     focus:Roo.emptyFn
30419 });
30420
30421 /**
30422  * @class Roo.Toolbar.Fill
30423  * @extends Roo.Toolbar.Spacer
30424  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30425  * @constructor
30426  * Creates a new Spacer
30427  */
30428 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30429     // private
30430     render : function(td){
30431         td.style.width = '100%';
30432         Roo.Toolbar.Fill.superclass.render.call(this, td);
30433     }
30434 });
30435
30436 /**
30437  * @class Roo.Toolbar.TextItem
30438  * @extends Roo.Toolbar.Item
30439  * A simple class that renders text directly into a toolbar.
30440  * @constructor
30441  * Creates a new TextItem
30442  * @param {String} text
30443  */
30444 Roo.Toolbar.TextItem = function(cfg){
30445     var  text = cfg || "";
30446     if (typeof(cfg) == 'object') {
30447         text = cfg.text || "";
30448     }  else {
30449         cfg = null;
30450     }
30451     var s = document.createElement("span");
30452     s.className = "ytb-text";
30453     s.innerHTML = text;
30454     if (cfg) {
30455         cfg.el  = s;
30456     }
30457     
30458     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30459 };
30460 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30461     
30462      
30463     enable:Roo.emptyFn,
30464     disable:Roo.emptyFn,
30465     focus:Roo.emptyFn
30466 });
30467
30468 /**
30469  * @class Roo.Toolbar.Button
30470  * @extends Roo.Button
30471  * A button that renders into a toolbar.
30472  * @constructor
30473  * Creates a new Button
30474  * @param {Object} config A standard {@link Roo.Button} config object
30475  */
30476 Roo.Toolbar.Button = function(config){
30477     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30478 };
30479 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30480     render : function(td){
30481         this.td = td;
30482         Roo.Toolbar.Button.superclass.render.call(this, td);
30483     },
30484     
30485     /**
30486      * Removes and destroys this button
30487      */
30488     destroy : function(){
30489         Roo.Toolbar.Button.superclass.destroy.call(this);
30490         this.td.parentNode.removeChild(this.td);
30491     },
30492     
30493     /**
30494      * Shows this button
30495      */
30496     show: function(){
30497         this.hidden = false;
30498         this.td.style.display = "";
30499     },
30500     
30501     /**
30502      * Hides this button
30503      */
30504     hide: function(){
30505         this.hidden = true;
30506         this.td.style.display = "none";
30507     },
30508
30509     /**
30510      * Disables this item
30511      */
30512     disable : function(){
30513         Roo.fly(this.td).addClass("x-item-disabled");
30514         this.disabled = true;
30515     },
30516
30517     /**
30518      * Enables this item
30519      */
30520     enable : function(){
30521         Roo.fly(this.td).removeClass("x-item-disabled");
30522         this.disabled = false;
30523     }
30524 });
30525 // backwards compat
30526 Roo.ToolbarButton = Roo.Toolbar.Button;
30527
30528 /**
30529  * @class Roo.Toolbar.SplitButton
30530  * @extends Roo.SplitButton
30531  * A menu button that renders into a toolbar.
30532  * @constructor
30533  * Creates a new SplitButton
30534  * @param {Object} config A standard {@link Roo.SplitButton} config object
30535  */
30536 Roo.Toolbar.SplitButton = function(config){
30537     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30538 };
30539 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30540     render : function(td){
30541         this.td = td;
30542         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30543     },
30544     
30545     /**
30546      * Removes and destroys this button
30547      */
30548     destroy : function(){
30549         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30550         this.td.parentNode.removeChild(this.td);
30551     },
30552     
30553     /**
30554      * Shows this button
30555      */
30556     show: function(){
30557         this.hidden = false;
30558         this.td.style.display = "";
30559     },
30560     
30561     /**
30562      * Hides this button
30563      */
30564     hide: function(){
30565         this.hidden = true;
30566         this.td.style.display = "none";
30567     }
30568 });
30569
30570 // backwards compat
30571 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30572  * Based on:
30573  * Ext JS Library 1.1.1
30574  * Copyright(c) 2006-2007, Ext JS, LLC.
30575  *
30576  * Originally Released Under LGPL - original licence link has changed is not relivant.
30577  *
30578  * Fork - LGPL
30579  * <script type="text/javascript">
30580  */
30581  
30582 /**
30583  * @class Roo.PagingToolbar
30584  * @extends Roo.Toolbar
30585  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30586  * @constructor
30587  * Create a new PagingToolbar
30588  * @param {Object} config The config object
30589  */
30590 Roo.PagingToolbar = function(el, ds, config)
30591 {
30592     // old args format still supported... - xtype is prefered..
30593     if (typeof(el) == 'object' && el.xtype) {
30594         // created from xtype...
30595         config = el;
30596         ds = el.dataSource;
30597         el = config.container;
30598     }
30599     var items = [];
30600     if (config.items) {
30601         items = config.items;
30602         config.items = [];
30603     }
30604     
30605     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30606     this.ds = ds;
30607     this.cursor = 0;
30608     this.renderButtons(this.el);
30609     this.bind(ds);
30610     
30611     // supprot items array.
30612    
30613     Roo.each(items, function(e) {
30614         this.add(Roo.factory(e));
30615     },this);
30616     
30617 };
30618
30619 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30620     /**
30621      * @cfg {Roo.data.Store} dataSource
30622      * The underlying data store providing the paged data
30623      */
30624     /**
30625      * @cfg {String/HTMLElement/Element} container
30626      * container The id or element that will contain the toolbar
30627      */
30628     /**
30629      * @cfg {Boolean} displayInfo
30630      * True to display the displayMsg (defaults to false)
30631      */
30632     /**
30633      * @cfg {Number} pageSize
30634      * The number of records to display per page (defaults to 20)
30635      */
30636     pageSize: 20,
30637     /**
30638      * @cfg {String} displayMsg
30639      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30640      */
30641     displayMsg : 'Displaying {0} - {1} of {2}',
30642     /**
30643      * @cfg {String} emptyMsg
30644      * The message to display when no records are found (defaults to "No data to display")
30645      */
30646     emptyMsg : 'No data to display',
30647     /**
30648      * Customizable piece of the default paging text (defaults to "Page")
30649      * @type String
30650      */
30651     beforePageText : "Page",
30652     /**
30653      * Customizable piece of the default paging text (defaults to "of %0")
30654      * @type String
30655      */
30656     afterPageText : "of {0}",
30657     /**
30658      * Customizable piece of the default paging text (defaults to "First Page")
30659      * @type String
30660      */
30661     firstText : "First Page",
30662     /**
30663      * Customizable piece of the default paging text (defaults to "Previous Page")
30664      * @type String
30665      */
30666     prevText : "Previous Page",
30667     /**
30668      * Customizable piece of the default paging text (defaults to "Next Page")
30669      * @type String
30670      */
30671     nextText : "Next Page",
30672     /**
30673      * Customizable piece of the default paging text (defaults to "Last Page")
30674      * @type String
30675      */
30676     lastText : "Last Page",
30677     /**
30678      * Customizable piece of the default paging text (defaults to "Refresh")
30679      * @type String
30680      */
30681     refreshText : "Refresh",
30682
30683     // private
30684     renderButtons : function(el){
30685         Roo.PagingToolbar.superclass.render.call(this, el);
30686         this.first = this.addButton({
30687             tooltip: this.firstText,
30688             cls: "x-btn-icon x-grid-page-first",
30689             disabled: true,
30690             handler: this.onClick.createDelegate(this, ["first"])
30691         });
30692         this.prev = this.addButton({
30693             tooltip: this.prevText,
30694             cls: "x-btn-icon x-grid-page-prev",
30695             disabled: true,
30696             handler: this.onClick.createDelegate(this, ["prev"])
30697         });
30698         //this.addSeparator();
30699         this.add(this.beforePageText);
30700         this.field = Roo.get(this.addDom({
30701            tag: "input",
30702            type: "text",
30703            size: "3",
30704            value: "1",
30705            cls: "x-grid-page-number"
30706         }).el);
30707         this.field.on("keydown", this.onPagingKeydown, this);
30708         this.field.on("focus", function(){this.dom.select();});
30709         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30710         this.field.setHeight(18);
30711         //this.addSeparator();
30712         this.next = this.addButton({
30713             tooltip: this.nextText,
30714             cls: "x-btn-icon x-grid-page-next",
30715             disabled: true,
30716             handler: this.onClick.createDelegate(this, ["next"])
30717         });
30718         this.last = this.addButton({
30719             tooltip: this.lastText,
30720             cls: "x-btn-icon x-grid-page-last",
30721             disabled: true,
30722             handler: this.onClick.createDelegate(this, ["last"])
30723         });
30724         //this.addSeparator();
30725         this.loading = this.addButton({
30726             tooltip: this.refreshText,
30727             cls: "x-btn-icon x-grid-loading",
30728             handler: this.onClick.createDelegate(this, ["refresh"])
30729         });
30730
30731         if(this.displayInfo){
30732             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30733         }
30734     },
30735
30736     // private
30737     updateInfo : function(){
30738         if(this.displayEl){
30739             var count = this.ds.getCount();
30740             var msg = count == 0 ?
30741                 this.emptyMsg :
30742                 String.format(
30743                     this.displayMsg,
30744                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30745                 );
30746             this.displayEl.update(msg);
30747         }
30748     },
30749
30750     // private
30751     onLoad : function(ds, r, o){
30752        this.cursor = o.params ? o.params.start : 0;
30753        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30754
30755        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30756        this.field.dom.value = ap;
30757        this.first.setDisabled(ap == 1);
30758        this.prev.setDisabled(ap == 1);
30759        this.next.setDisabled(ap == ps);
30760        this.last.setDisabled(ap == ps);
30761        this.loading.enable();
30762        this.updateInfo();
30763     },
30764
30765     // private
30766     getPageData : function(){
30767         var total = this.ds.getTotalCount();
30768         return {
30769             total : total,
30770             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30771             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30772         };
30773     },
30774
30775     // private
30776     onLoadError : function(){
30777         this.loading.enable();
30778     },
30779
30780     // private
30781     onPagingKeydown : function(e){
30782         var k = e.getKey();
30783         var d = this.getPageData();
30784         if(k == e.RETURN){
30785             var v = this.field.dom.value, pageNum;
30786             if(!v || isNaN(pageNum = parseInt(v, 10))){
30787                 this.field.dom.value = d.activePage;
30788                 return;
30789             }
30790             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30791             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30792             e.stopEvent();
30793         }
30794         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))
30795         {
30796           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30797           this.field.dom.value = pageNum;
30798           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30799           e.stopEvent();
30800         }
30801         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30802         {
30803           var v = this.field.dom.value, pageNum; 
30804           var increment = (e.shiftKey) ? 10 : 1;
30805           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30806             increment *= -1;
30807           }
30808           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30809             this.field.dom.value = d.activePage;
30810             return;
30811           }
30812           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30813           {
30814             this.field.dom.value = parseInt(v, 10) + increment;
30815             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30816             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30817           }
30818           e.stopEvent();
30819         }
30820     },
30821
30822     // private
30823     beforeLoad : function(){
30824         if(this.loading){
30825             this.loading.disable();
30826         }
30827     },
30828
30829     // private
30830     onClick : function(which){
30831         var ds = this.ds;
30832         switch(which){
30833             case "first":
30834                 ds.load({params:{start: 0, limit: this.pageSize}});
30835             break;
30836             case "prev":
30837                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30838             break;
30839             case "next":
30840                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30841             break;
30842             case "last":
30843                 var total = ds.getTotalCount();
30844                 var extra = total % this.pageSize;
30845                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30846                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30847             break;
30848             case "refresh":
30849                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30850             break;
30851         }
30852     },
30853
30854     /**
30855      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30856      * @param {Roo.data.Store} store The data store to unbind
30857      */
30858     unbind : function(ds){
30859         ds.un("beforeload", this.beforeLoad, this);
30860         ds.un("load", this.onLoad, this);
30861         ds.un("loadexception", this.onLoadError, this);
30862         ds.un("remove", this.updateInfo, this);
30863         ds.un("add", this.updateInfo, this);
30864         this.ds = undefined;
30865     },
30866
30867     /**
30868      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30869      * @param {Roo.data.Store} store The data store to bind
30870      */
30871     bind : function(ds){
30872         ds.on("beforeload", this.beforeLoad, this);
30873         ds.on("load", this.onLoad, this);
30874         ds.on("loadexception", this.onLoadError, this);
30875         ds.on("remove", this.updateInfo, this);
30876         ds.on("add", this.updateInfo, this);
30877         this.ds = ds;
30878     }
30879 });/*
30880  * Based on:
30881  * Ext JS Library 1.1.1
30882  * Copyright(c) 2006-2007, Ext JS, LLC.
30883  *
30884  * Originally Released Under LGPL - original licence link has changed is not relivant.
30885  *
30886  * Fork - LGPL
30887  * <script type="text/javascript">
30888  */
30889
30890 /**
30891  * @class Roo.Resizable
30892  * @extends Roo.util.Observable
30893  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30894  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30895  * 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
30896  * the element will be wrapped for you automatically.</p>
30897  * <p>Here is the list of valid resize handles:</p>
30898  * <pre>
30899 Value   Description
30900 ------  -------------------
30901  'n'     north
30902  's'     south
30903  'e'     east
30904  'w'     west
30905  'nw'    northwest
30906  'sw'    southwest
30907  'se'    southeast
30908  'ne'    northeast
30909  'hd'    horizontal drag
30910  'all'   all
30911 </pre>
30912  * <p>Here's an example showing the creation of a typical Resizable:</p>
30913  * <pre><code>
30914 var resizer = new Roo.Resizable("element-id", {
30915     handles: 'all',
30916     minWidth: 200,
30917     minHeight: 100,
30918     maxWidth: 500,
30919     maxHeight: 400,
30920     pinned: true
30921 });
30922 resizer.on("resize", myHandler);
30923 </code></pre>
30924  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30925  * resizer.east.setDisplayed(false);</p>
30926  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30927  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30928  * resize operation's new size (defaults to [0, 0])
30929  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30930  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30931  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30932  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30933  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30934  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30935  * @cfg {Number} width The width of the element in pixels (defaults to null)
30936  * @cfg {Number} height The height of the element in pixels (defaults to null)
30937  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30938  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30939  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30940  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30941  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30942  * in favor of the handles config option (defaults to false)
30943  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30944  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30945  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30946  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30947  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30948  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30949  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30950  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30951  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30952  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30953  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30954  * @constructor
30955  * Create a new resizable component
30956  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30957  * @param {Object} config configuration options
30958   */
30959 Roo.Resizable = function(el, config)
30960 {
30961     this.el = Roo.get(el);
30962
30963     if(config && config.wrap){
30964         config.resizeChild = this.el;
30965         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30966         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30967         this.el.setStyle("overflow", "hidden");
30968         this.el.setPositioning(config.resizeChild.getPositioning());
30969         config.resizeChild.clearPositioning();
30970         if(!config.width || !config.height){
30971             var csize = config.resizeChild.getSize();
30972             this.el.setSize(csize.width, csize.height);
30973         }
30974         if(config.pinned && !config.adjustments){
30975             config.adjustments = "auto";
30976         }
30977     }
30978
30979     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30980     this.proxy.unselectable();
30981     this.proxy.enableDisplayMode('block');
30982
30983     Roo.apply(this, config);
30984
30985     if(this.pinned){
30986         this.disableTrackOver = true;
30987         this.el.addClass("x-resizable-pinned");
30988     }
30989     // if the element isn't positioned, make it relative
30990     var position = this.el.getStyle("position");
30991     if(position != "absolute" && position != "fixed"){
30992         this.el.setStyle("position", "relative");
30993     }
30994     if(!this.handles){ // no handles passed, must be legacy style
30995         this.handles = 's,e,se';
30996         if(this.multiDirectional){
30997             this.handles += ',n,w';
30998         }
30999     }
31000     if(this.handles == "all"){
31001         this.handles = "n s e w ne nw se sw";
31002     }
31003     var hs = this.handles.split(/\s*?[,;]\s*?| /);
31004     var ps = Roo.Resizable.positions;
31005     for(var i = 0, len = hs.length; i < len; i++){
31006         if(hs[i] && ps[hs[i]]){
31007             var pos = ps[hs[i]];
31008             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
31009         }
31010     }
31011     // legacy
31012     this.corner = this.southeast;
31013     
31014     // updateBox = the box can move..
31015     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
31016         this.updateBox = true;
31017     }
31018
31019     this.activeHandle = null;
31020
31021     if(this.resizeChild){
31022         if(typeof this.resizeChild == "boolean"){
31023             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
31024         }else{
31025             this.resizeChild = Roo.get(this.resizeChild, true);
31026         }
31027     }
31028     
31029     if(this.adjustments == "auto"){
31030         var rc = this.resizeChild;
31031         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
31032         if(rc && (hw || hn)){
31033             rc.position("relative");
31034             rc.setLeft(hw ? hw.el.getWidth() : 0);
31035             rc.setTop(hn ? hn.el.getHeight() : 0);
31036         }
31037         this.adjustments = [
31038             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31039             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31040         ];
31041     }
31042
31043     if(this.draggable){
31044         this.dd = this.dynamic ?
31045             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31046         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31047     }
31048
31049     // public events
31050     this.addEvents({
31051         /**
31052          * @event beforeresize
31053          * Fired before resize is allowed. Set enabled to false to cancel resize.
31054          * @param {Roo.Resizable} this
31055          * @param {Roo.EventObject} e The mousedown event
31056          */
31057         "beforeresize" : true,
31058         /**
31059          * @event resizing
31060          * Fired a resizing.
31061          * @param {Roo.Resizable} this
31062          * @param {Number} x The new x position
31063          * @param {Number} y The new y position
31064          * @param {Number} w The new w width
31065          * @param {Number} h The new h hight
31066          * @param {Roo.EventObject} e The mouseup event
31067          */
31068         "resizing" : true,
31069         /**
31070          * @event resize
31071          * Fired after a resize.
31072          * @param {Roo.Resizable} this
31073          * @param {Number} width The new width
31074          * @param {Number} height The new height
31075          * @param {Roo.EventObject} e The mouseup event
31076          */
31077         "resize" : true
31078     });
31079
31080     if(this.width !== null && this.height !== null){
31081         this.resizeTo(this.width, this.height);
31082     }else{
31083         this.updateChildSize();
31084     }
31085     if(Roo.isIE){
31086         this.el.dom.style.zoom = 1;
31087     }
31088     Roo.Resizable.superclass.constructor.call(this);
31089 };
31090
31091 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31092         resizeChild : false,
31093         adjustments : [0, 0],
31094         minWidth : 5,
31095         minHeight : 5,
31096         maxWidth : 10000,
31097         maxHeight : 10000,
31098         enabled : true,
31099         animate : false,
31100         duration : .35,
31101         dynamic : false,
31102         handles : false,
31103         multiDirectional : false,
31104         disableTrackOver : false,
31105         easing : 'easeOutStrong',
31106         widthIncrement : 0,
31107         heightIncrement : 0,
31108         pinned : false,
31109         width : null,
31110         height : null,
31111         preserveRatio : false,
31112         transparent: false,
31113         minX: 0,
31114         minY: 0,
31115         draggable: false,
31116
31117         /**
31118          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31119          */
31120         constrainTo: undefined,
31121         /**
31122          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31123          */
31124         resizeRegion: undefined,
31125
31126
31127     /**
31128      * Perform a manual resize
31129      * @param {Number} width
31130      * @param {Number} height
31131      */
31132     resizeTo : function(width, height){
31133         this.el.setSize(width, height);
31134         this.updateChildSize();
31135         this.fireEvent("resize", this, width, height, null);
31136     },
31137
31138     // private
31139     startSizing : function(e, handle){
31140         this.fireEvent("beforeresize", this, e);
31141         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31142
31143             if(!this.overlay){
31144                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31145                 this.overlay.unselectable();
31146                 this.overlay.enableDisplayMode("block");
31147                 this.overlay.on("mousemove", this.onMouseMove, this);
31148                 this.overlay.on("mouseup", this.onMouseUp, this);
31149             }
31150             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31151
31152             this.resizing = true;
31153             this.startBox = this.el.getBox();
31154             this.startPoint = e.getXY();
31155             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31156                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31157
31158             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31159             this.overlay.show();
31160
31161             if(this.constrainTo) {
31162                 var ct = Roo.get(this.constrainTo);
31163                 this.resizeRegion = ct.getRegion().adjust(
31164                     ct.getFrameWidth('t'),
31165                     ct.getFrameWidth('l'),
31166                     -ct.getFrameWidth('b'),
31167                     -ct.getFrameWidth('r')
31168                 );
31169             }
31170
31171             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31172             this.proxy.show();
31173             this.proxy.setBox(this.startBox);
31174             if(!this.dynamic){
31175                 this.proxy.setStyle('visibility', 'visible');
31176             }
31177         }
31178     },
31179
31180     // private
31181     onMouseDown : function(handle, e){
31182         if(this.enabled){
31183             e.stopEvent();
31184             this.activeHandle = handle;
31185             this.startSizing(e, handle);
31186         }
31187     },
31188
31189     // private
31190     onMouseUp : function(e){
31191         var size = this.resizeElement();
31192         this.resizing = false;
31193         this.handleOut();
31194         this.overlay.hide();
31195         this.proxy.hide();
31196         this.fireEvent("resize", this, size.width, size.height, e);
31197     },
31198
31199     // private
31200     updateChildSize : function(){
31201         
31202         if(this.resizeChild){
31203             var el = this.el;
31204             var child = this.resizeChild;
31205             var adj = this.adjustments;
31206             if(el.dom.offsetWidth){
31207                 var b = el.getSize(true);
31208                 child.setSize(b.width+adj[0], b.height+adj[1]);
31209             }
31210             // Second call here for IE
31211             // The first call enables instant resizing and
31212             // the second call corrects scroll bars if they
31213             // exist
31214             if(Roo.isIE){
31215                 setTimeout(function(){
31216                     if(el.dom.offsetWidth){
31217                         var b = el.getSize(true);
31218                         child.setSize(b.width+adj[0], b.height+adj[1]);
31219                     }
31220                 }, 10);
31221             }
31222         }
31223     },
31224
31225     // private
31226     snap : function(value, inc, min){
31227         if(!inc || !value) {
31228             return value;
31229         }
31230         var newValue = value;
31231         var m = value % inc;
31232         if(m > 0){
31233             if(m > (inc/2)){
31234                 newValue = value + (inc-m);
31235             }else{
31236                 newValue = value - m;
31237             }
31238         }
31239         return Math.max(min, newValue);
31240     },
31241
31242     // private
31243     resizeElement : function(){
31244         var box = this.proxy.getBox();
31245         if(this.updateBox){
31246             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31247         }else{
31248             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31249         }
31250         this.updateChildSize();
31251         if(!this.dynamic){
31252             this.proxy.hide();
31253         }
31254         return box;
31255     },
31256
31257     // private
31258     constrain : function(v, diff, m, mx){
31259         if(v - diff < m){
31260             diff = v - m;
31261         }else if(v - diff > mx){
31262             diff = mx - v;
31263         }
31264         return diff;
31265     },
31266
31267     // private
31268     onMouseMove : function(e){
31269         
31270         if(this.enabled){
31271             try{// try catch so if something goes wrong the user doesn't get hung
31272
31273             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31274                 return;
31275             }
31276
31277             //var curXY = this.startPoint;
31278             var curSize = this.curSize || this.startBox;
31279             var x = this.startBox.x, y = this.startBox.y;
31280             var ox = x, oy = y;
31281             var w = curSize.width, h = curSize.height;
31282             var ow = w, oh = h;
31283             var mw = this.minWidth, mh = this.minHeight;
31284             var mxw = this.maxWidth, mxh = this.maxHeight;
31285             var wi = this.widthIncrement;
31286             var hi = this.heightIncrement;
31287
31288             var eventXY = e.getXY();
31289             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31290             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31291
31292             var pos = this.activeHandle.position;
31293
31294             switch(pos){
31295                 case "east":
31296                     w += diffX;
31297                     w = Math.min(Math.max(mw, w), mxw);
31298                     break;
31299              
31300                 case "south":
31301                     h += diffY;
31302                     h = Math.min(Math.max(mh, h), mxh);
31303                     break;
31304                 case "southeast":
31305                     w += diffX;
31306                     h += diffY;
31307                     w = Math.min(Math.max(mw, w), mxw);
31308                     h = Math.min(Math.max(mh, h), mxh);
31309                     break;
31310                 case "north":
31311                     diffY = this.constrain(h, diffY, mh, mxh);
31312                     y += diffY;
31313                     h -= diffY;
31314                     break;
31315                 case "hdrag":
31316                     
31317                     if (wi) {
31318                         var adiffX = Math.abs(diffX);
31319                         var sub = (adiffX % wi); // how much 
31320                         if (sub > (wi/2)) { // far enough to snap
31321                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31322                         } else {
31323                             // remove difference.. 
31324                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31325                         }
31326                     }
31327                     x += diffX;
31328                     x = Math.max(this.minX, x);
31329                     break;
31330                 case "west":
31331                     diffX = this.constrain(w, diffX, mw, mxw);
31332                     x += diffX;
31333                     w -= diffX;
31334                     break;
31335                 case "northeast":
31336                     w += diffX;
31337                     w = Math.min(Math.max(mw, w), mxw);
31338                     diffY = this.constrain(h, diffY, mh, mxh);
31339                     y += diffY;
31340                     h -= diffY;
31341                     break;
31342                 case "northwest":
31343                     diffX = this.constrain(w, diffX, mw, mxw);
31344                     diffY = this.constrain(h, diffY, mh, mxh);
31345                     y += diffY;
31346                     h -= diffY;
31347                     x += diffX;
31348                     w -= diffX;
31349                     break;
31350                case "southwest":
31351                     diffX = this.constrain(w, diffX, mw, mxw);
31352                     h += diffY;
31353                     h = Math.min(Math.max(mh, h), mxh);
31354                     x += diffX;
31355                     w -= diffX;
31356                     break;
31357             }
31358
31359             var sw = this.snap(w, wi, mw);
31360             var sh = this.snap(h, hi, mh);
31361             if(sw != w || sh != h){
31362                 switch(pos){
31363                     case "northeast":
31364                         y -= sh - h;
31365                     break;
31366                     case "north":
31367                         y -= sh - h;
31368                         break;
31369                     case "southwest":
31370                         x -= sw - w;
31371                     break;
31372                     case "west":
31373                         x -= sw - w;
31374                         break;
31375                     case "northwest":
31376                         x -= sw - w;
31377                         y -= sh - h;
31378                     break;
31379                 }
31380                 w = sw;
31381                 h = sh;
31382             }
31383
31384             if(this.preserveRatio){
31385                 switch(pos){
31386                     case "southeast":
31387                     case "east":
31388                         h = oh * (w/ow);
31389                         h = Math.min(Math.max(mh, h), mxh);
31390                         w = ow * (h/oh);
31391                        break;
31392                     case "south":
31393                         w = ow * (h/oh);
31394                         w = Math.min(Math.max(mw, w), mxw);
31395                         h = oh * (w/ow);
31396                         break;
31397                     case "northeast":
31398                         w = ow * (h/oh);
31399                         w = Math.min(Math.max(mw, w), mxw);
31400                         h = oh * (w/ow);
31401                     break;
31402                     case "north":
31403                         var tw = w;
31404                         w = ow * (h/oh);
31405                         w = Math.min(Math.max(mw, w), mxw);
31406                         h = oh * (w/ow);
31407                         x += (tw - w) / 2;
31408                         break;
31409                     case "southwest":
31410                         h = oh * (w/ow);
31411                         h = Math.min(Math.max(mh, h), mxh);
31412                         var tw = w;
31413                         w = ow * (h/oh);
31414                         x += tw - w;
31415                         break;
31416                     case "west":
31417                         var th = h;
31418                         h = oh * (w/ow);
31419                         h = Math.min(Math.max(mh, h), mxh);
31420                         y += (th - h) / 2;
31421                         var tw = w;
31422                         w = ow * (h/oh);
31423                         x += tw - w;
31424                        break;
31425                     case "northwest":
31426                         var tw = w;
31427                         var th = h;
31428                         h = oh * (w/ow);
31429                         h = Math.min(Math.max(mh, h), mxh);
31430                         w = ow * (h/oh);
31431                         y += th - h;
31432                         x += tw - w;
31433                        break;
31434
31435                 }
31436             }
31437             if (pos == 'hdrag') {
31438                 w = ow;
31439             }
31440             this.proxy.setBounds(x, y, w, h);
31441             if(this.dynamic){
31442                 this.resizeElement();
31443             }
31444             }catch(e){}
31445         }
31446         this.fireEvent("resizing", this, x, y, w, h, e);
31447     },
31448
31449     // private
31450     handleOver : function(){
31451         if(this.enabled){
31452             this.el.addClass("x-resizable-over");
31453         }
31454     },
31455
31456     // private
31457     handleOut : function(){
31458         if(!this.resizing){
31459             this.el.removeClass("x-resizable-over");
31460         }
31461     },
31462
31463     /**
31464      * Returns the element this component is bound to.
31465      * @return {Roo.Element}
31466      */
31467     getEl : function(){
31468         return this.el;
31469     },
31470
31471     /**
31472      * Returns the resizeChild element (or null).
31473      * @return {Roo.Element}
31474      */
31475     getResizeChild : function(){
31476         return this.resizeChild;
31477     },
31478     groupHandler : function()
31479     {
31480         
31481     },
31482     /**
31483      * Destroys this resizable. If the element was wrapped and
31484      * removeEl is not true then the element remains.
31485      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31486      */
31487     destroy : function(removeEl){
31488         this.proxy.remove();
31489         if(this.overlay){
31490             this.overlay.removeAllListeners();
31491             this.overlay.remove();
31492         }
31493         var ps = Roo.Resizable.positions;
31494         for(var k in ps){
31495             if(typeof ps[k] != "function" && this[ps[k]]){
31496                 var h = this[ps[k]];
31497                 h.el.removeAllListeners();
31498                 h.el.remove();
31499             }
31500         }
31501         if(removeEl){
31502             this.el.update("");
31503             this.el.remove();
31504         }
31505     }
31506 });
31507
31508 // private
31509 // hash to map config positions to true positions
31510 Roo.Resizable.positions = {
31511     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31512     hd: "hdrag"
31513 };
31514
31515 // private
31516 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31517     if(!this.tpl){
31518         // only initialize the template if resizable is used
31519         var tpl = Roo.DomHelper.createTemplate(
31520             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31521         );
31522         tpl.compile();
31523         Roo.Resizable.Handle.prototype.tpl = tpl;
31524     }
31525     this.position = pos;
31526     this.rz = rz;
31527     // show north drag fro topdra
31528     var handlepos = pos == 'hdrag' ? 'north' : pos;
31529     
31530     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31531     if (pos == 'hdrag') {
31532         this.el.setStyle('cursor', 'pointer');
31533     }
31534     this.el.unselectable();
31535     if(transparent){
31536         this.el.setOpacity(0);
31537     }
31538     this.el.on("mousedown", this.onMouseDown, this);
31539     if(!disableTrackOver){
31540         this.el.on("mouseover", this.onMouseOver, this);
31541         this.el.on("mouseout", this.onMouseOut, this);
31542     }
31543 };
31544
31545 // private
31546 Roo.Resizable.Handle.prototype = {
31547     afterResize : function(rz){
31548         Roo.log('after?');
31549         // do nothing
31550     },
31551     // private
31552     onMouseDown : function(e){
31553         this.rz.onMouseDown(this, e);
31554     },
31555     // private
31556     onMouseOver : function(e){
31557         this.rz.handleOver(this, e);
31558     },
31559     // private
31560     onMouseOut : function(e){
31561         this.rz.handleOut(this, e);
31562     }
31563 };/*
31564  * Based on:
31565  * Ext JS Library 1.1.1
31566  * Copyright(c) 2006-2007, Ext JS, LLC.
31567  *
31568  * Originally Released Under LGPL - original licence link has changed is not relivant.
31569  *
31570  * Fork - LGPL
31571  * <script type="text/javascript">
31572  */
31573
31574 /**
31575  * @class Roo.Editor
31576  * @extends Roo.Component
31577  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31578  * @constructor
31579  * Create a new Editor
31580  * @param {Roo.form.Field} field The Field object (or descendant)
31581  * @param {Object} config The config object
31582  */
31583 Roo.Editor = function(field, config){
31584     Roo.Editor.superclass.constructor.call(this, config);
31585     this.field = field;
31586     this.addEvents({
31587         /**
31588              * @event beforestartedit
31589              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31590              * false from the handler of this event.
31591              * @param {Editor} this
31592              * @param {Roo.Element} boundEl The underlying element bound to this editor
31593              * @param {Mixed} value The field value being set
31594              */
31595         "beforestartedit" : true,
31596         /**
31597              * @event startedit
31598              * Fires when this editor is displayed
31599              * @param {Roo.Element} boundEl The underlying element bound to this editor
31600              * @param {Mixed} value The starting field value
31601              */
31602         "startedit" : true,
31603         /**
31604              * @event beforecomplete
31605              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31606              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31607              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31608              * event will not fire since no edit actually occurred.
31609              * @param {Editor} this
31610              * @param {Mixed} value The current field value
31611              * @param {Mixed} startValue The original field value
31612              */
31613         "beforecomplete" : true,
31614         /**
31615              * @event complete
31616              * Fires after editing is complete and any changed value has been written to the underlying field.
31617              * @param {Editor} this
31618              * @param {Mixed} value The current field value
31619              * @param {Mixed} startValue The original field value
31620              */
31621         "complete" : true,
31622         /**
31623          * @event specialkey
31624          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31625          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31626          * @param {Roo.form.Field} this
31627          * @param {Roo.EventObject} e The event object
31628          */
31629         "specialkey" : true
31630     });
31631 };
31632
31633 Roo.extend(Roo.Editor, Roo.Component, {
31634     /**
31635      * @cfg {Boolean/String} autosize
31636      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31637      * or "height" to adopt the height only (defaults to false)
31638      */
31639     /**
31640      * @cfg {Boolean} revertInvalid
31641      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31642      * validation fails (defaults to true)
31643      */
31644     /**
31645      * @cfg {Boolean} ignoreNoChange
31646      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31647      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31648      * will never be ignored.
31649      */
31650     /**
31651      * @cfg {Boolean} hideEl
31652      * False to keep the bound element visible while the editor is displayed (defaults to true)
31653      */
31654     /**
31655      * @cfg {Mixed} value
31656      * The data value of the underlying field (defaults to "")
31657      */
31658     value : "",
31659     /**
31660      * @cfg {String} alignment
31661      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31662      */
31663     alignment: "c-c?",
31664     /**
31665      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31666      * for bottom-right shadow (defaults to "frame")
31667      */
31668     shadow : "frame",
31669     /**
31670      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31671      */
31672     constrain : false,
31673     /**
31674      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31675      */
31676     completeOnEnter : false,
31677     /**
31678      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31679      */
31680     cancelOnEsc : false,
31681     /**
31682      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31683      */
31684     updateEl : false,
31685
31686     // private
31687     onRender : function(ct, position){
31688         this.el = new Roo.Layer({
31689             shadow: this.shadow,
31690             cls: "x-editor",
31691             parentEl : ct,
31692             shim : this.shim,
31693             shadowOffset:4,
31694             id: this.id,
31695             constrain: this.constrain
31696         });
31697         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31698         if(this.field.msgTarget != 'title'){
31699             this.field.msgTarget = 'qtip';
31700         }
31701         this.field.render(this.el);
31702         if(Roo.isGecko){
31703             this.field.el.dom.setAttribute('autocomplete', 'off');
31704         }
31705         this.field.on("specialkey", this.onSpecialKey, this);
31706         if(this.swallowKeys){
31707             this.field.el.swallowEvent(['keydown','keypress']);
31708         }
31709         this.field.show();
31710         this.field.on("blur", this.onBlur, this);
31711         if(this.field.grow){
31712             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31713         }
31714     },
31715
31716     onSpecialKey : function(field, e)
31717     {
31718         //Roo.log('editor onSpecialKey');
31719         if(this.completeOnEnter && e.getKey() == e.ENTER){
31720             e.stopEvent();
31721             this.completeEdit();
31722             return;
31723         }
31724         // do not fire special key otherwise it might hide close the editor...
31725         if(e.getKey() == e.ENTER){    
31726             return;
31727         }
31728         if(this.cancelOnEsc && e.getKey() == e.ESC){
31729             this.cancelEdit();
31730             return;
31731         } 
31732         this.fireEvent('specialkey', field, e);
31733     
31734     },
31735
31736     /**
31737      * Starts the editing process and shows the editor.
31738      * @param {String/HTMLElement/Element} el The element to edit
31739      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31740       * to the innerHTML of el.
31741      */
31742     startEdit : function(el, value){
31743         if(this.editing){
31744             this.completeEdit();
31745         }
31746         this.boundEl = Roo.get(el);
31747         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31748         if(!this.rendered){
31749             this.render(this.parentEl || document.body);
31750         }
31751         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31752             return;
31753         }
31754         this.startValue = v;
31755         this.field.setValue(v);
31756         if(this.autoSize){
31757             var sz = this.boundEl.getSize();
31758             switch(this.autoSize){
31759                 case "width":
31760                 this.setSize(sz.width,  "");
31761                 break;
31762                 case "height":
31763                 this.setSize("",  sz.height);
31764                 break;
31765                 default:
31766                 this.setSize(sz.width,  sz.height);
31767             }
31768         }
31769         this.el.alignTo(this.boundEl, this.alignment);
31770         this.editing = true;
31771         if(Roo.QuickTips){
31772             Roo.QuickTips.disable();
31773         }
31774         this.show();
31775     },
31776
31777     /**
31778      * Sets the height and width of this editor.
31779      * @param {Number} width The new width
31780      * @param {Number} height The new height
31781      */
31782     setSize : function(w, h){
31783         this.field.setSize(w, h);
31784         if(this.el){
31785             this.el.sync();
31786         }
31787     },
31788
31789     /**
31790      * Realigns the editor to the bound field based on the current alignment config value.
31791      */
31792     realign : function(){
31793         this.el.alignTo(this.boundEl, this.alignment);
31794     },
31795
31796     /**
31797      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31798      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31799      */
31800     completeEdit : function(remainVisible){
31801         if(!this.editing){
31802             return;
31803         }
31804         var v = this.getValue();
31805         if(this.revertInvalid !== false && !this.field.isValid()){
31806             v = this.startValue;
31807             this.cancelEdit(true);
31808         }
31809         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31810             this.editing = false;
31811             this.hide();
31812             return;
31813         }
31814         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31815             this.editing = false;
31816             if(this.updateEl && this.boundEl){
31817                 this.boundEl.update(v);
31818             }
31819             if(remainVisible !== true){
31820                 this.hide();
31821             }
31822             this.fireEvent("complete", this, v, this.startValue);
31823         }
31824     },
31825
31826     // private
31827     onShow : function(){
31828         this.el.show();
31829         if(this.hideEl !== false){
31830             this.boundEl.hide();
31831         }
31832         this.field.show();
31833         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31834             this.fixIEFocus = true;
31835             this.deferredFocus.defer(50, this);
31836         }else{
31837             this.field.focus();
31838         }
31839         this.fireEvent("startedit", this.boundEl, this.startValue);
31840     },
31841
31842     deferredFocus : function(){
31843         if(this.editing){
31844             this.field.focus();
31845         }
31846     },
31847
31848     /**
31849      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31850      * reverted to the original starting value.
31851      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31852      * cancel (defaults to false)
31853      */
31854     cancelEdit : function(remainVisible){
31855         if(this.editing){
31856             this.setValue(this.startValue);
31857             if(remainVisible !== true){
31858                 this.hide();
31859             }
31860         }
31861     },
31862
31863     // private
31864     onBlur : function(){
31865         if(this.allowBlur !== true && this.editing){
31866             this.completeEdit();
31867         }
31868     },
31869
31870     // private
31871     onHide : function(){
31872         if(this.editing){
31873             this.completeEdit();
31874             return;
31875         }
31876         this.field.blur();
31877         if(this.field.collapse){
31878             this.field.collapse();
31879         }
31880         this.el.hide();
31881         if(this.hideEl !== false){
31882             this.boundEl.show();
31883         }
31884         if(Roo.QuickTips){
31885             Roo.QuickTips.enable();
31886         }
31887     },
31888
31889     /**
31890      * Sets the data value of the editor
31891      * @param {Mixed} value Any valid value supported by the underlying field
31892      */
31893     setValue : function(v){
31894         this.field.setValue(v);
31895     },
31896
31897     /**
31898      * Gets the data value of the editor
31899      * @return {Mixed} The data value
31900      */
31901     getValue : function(){
31902         return this.field.getValue();
31903     }
31904 });/*
31905  * Based on:
31906  * Ext JS Library 1.1.1
31907  * Copyright(c) 2006-2007, Ext JS, LLC.
31908  *
31909  * Originally Released Under LGPL - original licence link has changed is not relivant.
31910  *
31911  * Fork - LGPL
31912  * <script type="text/javascript">
31913  */
31914  
31915 /**
31916  * @class Roo.BasicDialog
31917  * @extends Roo.util.Observable
31918  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31919  * <pre><code>
31920 var dlg = new Roo.BasicDialog("my-dlg", {
31921     height: 200,
31922     width: 300,
31923     minHeight: 100,
31924     minWidth: 150,
31925     modal: true,
31926     proxyDrag: true,
31927     shadow: true
31928 });
31929 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31930 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31931 dlg.addButton('Cancel', dlg.hide, dlg);
31932 dlg.show();
31933 </code></pre>
31934   <b>A Dialog should always be a direct child of the body element.</b>
31935  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31936  * @cfg {String} title Default text to display in the title bar (defaults to null)
31937  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31938  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31939  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31940  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31941  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31942  * (defaults to null with no animation)
31943  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31944  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31945  * property for valid values (defaults to 'all')
31946  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31947  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31948  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31949  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31950  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31951  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31952  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31953  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31954  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31955  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31956  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31957  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31958  * draggable = true (defaults to false)
31959  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31960  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31961  * shadow (defaults to false)
31962  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31963  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31964  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31965  * @cfg {Array} buttons Array of buttons
31966  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31967  * @constructor
31968  * Create a new BasicDialog.
31969  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31970  * @param {Object} config Configuration options
31971  */
31972 Roo.BasicDialog = function(el, config){
31973     this.el = Roo.get(el);
31974     var dh = Roo.DomHelper;
31975     if(!this.el && config && config.autoCreate){
31976         if(typeof config.autoCreate == "object"){
31977             if(!config.autoCreate.id){
31978                 config.autoCreate.id = el;
31979             }
31980             this.el = dh.append(document.body,
31981                         config.autoCreate, true);
31982         }else{
31983             this.el = dh.append(document.body,
31984                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31985         }
31986     }
31987     el = this.el;
31988     el.setDisplayed(true);
31989     el.hide = this.hideAction;
31990     this.id = el.id;
31991     el.addClass("x-dlg");
31992
31993     Roo.apply(this, config);
31994
31995     this.proxy = el.createProxy("x-dlg-proxy");
31996     this.proxy.hide = this.hideAction;
31997     this.proxy.setOpacity(.5);
31998     this.proxy.hide();
31999
32000     if(config.width){
32001         el.setWidth(config.width);
32002     }
32003     if(config.height){
32004         el.setHeight(config.height);
32005     }
32006     this.size = el.getSize();
32007     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
32008         this.xy = [config.x,config.y];
32009     }else{
32010         this.xy = el.getCenterXY(true);
32011     }
32012     /** The header element @type Roo.Element */
32013     this.header = el.child("> .x-dlg-hd");
32014     /** The body element @type Roo.Element */
32015     this.body = el.child("> .x-dlg-bd");
32016     /** The footer element @type Roo.Element */
32017     this.footer = el.child("> .x-dlg-ft");
32018
32019     if(!this.header){
32020         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
32021     }
32022     if(!this.body){
32023         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
32024     }
32025
32026     this.header.unselectable();
32027     if(this.title){
32028         this.header.update(this.title);
32029     }
32030     // this element allows the dialog to be focused for keyboard event
32031     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
32032     this.focusEl.swallowEvent("click", true);
32033
32034     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
32035
32036     // wrap the body and footer for special rendering
32037     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32038     if(this.footer){
32039         this.bwrap.dom.appendChild(this.footer.dom);
32040     }
32041
32042     this.bg = this.el.createChild({
32043         tag: "div", cls:"x-dlg-bg",
32044         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32045     });
32046     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32047
32048
32049     if(this.autoScroll !== false && !this.autoTabs){
32050         this.body.setStyle("overflow", "auto");
32051     }
32052
32053     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32054
32055     if(this.closable !== false){
32056         this.el.addClass("x-dlg-closable");
32057         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32058         this.close.on("click", this.closeClick, this);
32059         this.close.addClassOnOver("x-dlg-close-over");
32060     }
32061     if(this.collapsible !== false){
32062         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32063         this.collapseBtn.on("click", this.collapseClick, this);
32064         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32065         this.header.on("dblclick", this.collapseClick, this);
32066     }
32067     if(this.resizable !== false){
32068         this.el.addClass("x-dlg-resizable");
32069         this.resizer = new Roo.Resizable(el, {
32070             minWidth: this.minWidth || 80,
32071             minHeight:this.minHeight || 80,
32072             handles: this.resizeHandles || "all",
32073             pinned: true
32074         });
32075         this.resizer.on("beforeresize", this.beforeResize, this);
32076         this.resizer.on("resize", this.onResize, this);
32077     }
32078     if(this.draggable !== false){
32079         el.addClass("x-dlg-draggable");
32080         if (!this.proxyDrag) {
32081             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32082         }
32083         else {
32084             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32085         }
32086         dd.setHandleElId(this.header.id);
32087         dd.endDrag = this.endMove.createDelegate(this);
32088         dd.startDrag = this.startMove.createDelegate(this);
32089         dd.onDrag = this.onDrag.createDelegate(this);
32090         dd.scroll = false;
32091         this.dd = dd;
32092     }
32093     if(this.modal){
32094         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32095         this.mask.enableDisplayMode("block");
32096         this.mask.hide();
32097         this.el.addClass("x-dlg-modal");
32098     }
32099     if(this.shadow){
32100         this.shadow = new Roo.Shadow({
32101             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32102             offset : this.shadowOffset
32103         });
32104     }else{
32105         this.shadowOffset = 0;
32106     }
32107     if(Roo.useShims && this.shim !== false){
32108         this.shim = this.el.createShim();
32109         this.shim.hide = this.hideAction;
32110         this.shim.hide();
32111     }else{
32112         this.shim = false;
32113     }
32114     if(this.autoTabs){
32115         this.initTabs();
32116     }
32117     if (this.buttons) { 
32118         var bts= this.buttons;
32119         this.buttons = [];
32120         Roo.each(bts, function(b) {
32121             this.addButton(b);
32122         }, this);
32123     }
32124     
32125     
32126     this.addEvents({
32127         /**
32128          * @event keydown
32129          * Fires when a key is pressed
32130          * @param {Roo.BasicDialog} this
32131          * @param {Roo.EventObject} e
32132          */
32133         "keydown" : true,
32134         /**
32135          * @event move
32136          * Fires when this dialog is moved by the user.
32137          * @param {Roo.BasicDialog} this
32138          * @param {Number} x The new page X
32139          * @param {Number} y The new page Y
32140          */
32141         "move" : true,
32142         /**
32143          * @event resize
32144          * Fires when this dialog is resized by the user.
32145          * @param {Roo.BasicDialog} this
32146          * @param {Number} width The new width
32147          * @param {Number} height The new height
32148          */
32149         "resize" : true,
32150         /**
32151          * @event beforehide
32152          * Fires before this dialog is hidden.
32153          * @param {Roo.BasicDialog} this
32154          */
32155         "beforehide" : true,
32156         /**
32157          * @event hide
32158          * Fires when this dialog is hidden.
32159          * @param {Roo.BasicDialog} this
32160          */
32161         "hide" : true,
32162         /**
32163          * @event beforeshow
32164          * Fires before this dialog is shown.
32165          * @param {Roo.BasicDialog} this
32166          */
32167         "beforeshow" : true,
32168         /**
32169          * @event show
32170          * Fires when this dialog is shown.
32171          * @param {Roo.BasicDialog} this
32172          */
32173         "show" : true
32174     });
32175     el.on("keydown", this.onKeyDown, this);
32176     el.on("mousedown", this.toFront, this);
32177     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32178     this.el.hide();
32179     Roo.DialogManager.register(this);
32180     Roo.BasicDialog.superclass.constructor.call(this);
32181 };
32182
32183 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32184     shadowOffset: Roo.isIE ? 6 : 5,
32185     minHeight: 80,
32186     minWidth: 200,
32187     minButtonWidth: 75,
32188     defaultButton: null,
32189     buttonAlign: "right",
32190     tabTag: 'div',
32191     firstShow: true,
32192
32193     /**
32194      * Sets the dialog title text
32195      * @param {String} text The title text to display
32196      * @return {Roo.BasicDialog} this
32197      */
32198     setTitle : function(text){
32199         this.header.update(text);
32200         return this;
32201     },
32202
32203     // private
32204     closeClick : function(){
32205         this.hide();
32206     },
32207
32208     // private
32209     collapseClick : function(){
32210         this[this.collapsed ? "expand" : "collapse"]();
32211     },
32212
32213     /**
32214      * Collapses the dialog to its minimized state (only the title bar is visible).
32215      * Equivalent to the user clicking the collapse dialog button.
32216      */
32217     collapse : function(){
32218         if(!this.collapsed){
32219             this.collapsed = true;
32220             this.el.addClass("x-dlg-collapsed");
32221             this.restoreHeight = this.el.getHeight();
32222             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32223         }
32224     },
32225
32226     /**
32227      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32228      * clicking the expand dialog button.
32229      */
32230     expand : function(){
32231         if(this.collapsed){
32232             this.collapsed = false;
32233             this.el.removeClass("x-dlg-collapsed");
32234             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32235         }
32236     },
32237
32238     /**
32239      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32240      * @return {Roo.TabPanel} The tabs component
32241      */
32242     initTabs : function(){
32243         var tabs = this.getTabs();
32244         while(tabs.getTab(0)){
32245             tabs.removeTab(0);
32246         }
32247         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32248             var dom = el.dom;
32249             tabs.addTab(Roo.id(dom), dom.title);
32250             dom.title = "";
32251         });
32252         tabs.activate(0);
32253         return tabs;
32254     },
32255
32256     // private
32257     beforeResize : function(){
32258         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32259     },
32260
32261     // private
32262     onResize : function(){
32263         this.refreshSize();
32264         this.syncBodyHeight();
32265         this.adjustAssets();
32266         this.focus();
32267         this.fireEvent("resize", this, this.size.width, this.size.height);
32268     },
32269
32270     // private
32271     onKeyDown : function(e){
32272         if(this.isVisible()){
32273             this.fireEvent("keydown", this, e);
32274         }
32275     },
32276
32277     /**
32278      * Resizes the dialog.
32279      * @param {Number} width
32280      * @param {Number} height
32281      * @return {Roo.BasicDialog} this
32282      */
32283     resizeTo : function(width, height){
32284         this.el.setSize(width, height);
32285         this.size = {width: width, height: height};
32286         this.syncBodyHeight();
32287         if(this.fixedcenter){
32288             this.center();
32289         }
32290         if(this.isVisible()){
32291             this.constrainXY();
32292             this.adjustAssets();
32293         }
32294         this.fireEvent("resize", this, width, height);
32295         return this;
32296     },
32297
32298
32299     /**
32300      * Resizes the dialog to fit the specified content size.
32301      * @param {Number} width
32302      * @param {Number} height
32303      * @return {Roo.BasicDialog} this
32304      */
32305     setContentSize : function(w, h){
32306         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32307         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32308         //if(!this.el.isBorderBox()){
32309             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32310             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32311         //}
32312         if(this.tabs){
32313             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32314             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32315         }
32316         this.resizeTo(w, h);
32317         return this;
32318     },
32319
32320     /**
32321      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32322      * executed in response to a particular key being pressed while the dialog is active.
32323      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32324      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32325      * @param {Function} fn The function to call
32326      * @param {Object} scope (optional) The scope of the function
32327      * @return {Roo.BasicDialog} this
32328      */
32329     addKeyListener : function(key, fn, scope){
32330         var keyCode, shift, ctrl, alt;
32331         if(typeof key == "object" && !(key instanceof Array)){
32332             keyCode = key["key"];
32333             shift = key["shift"];
32334             ctrl = key["ctrl"];
32335             alt = key["alt"];
32336         }else{
32337             keyCode = key;
32338         }
32339         var handler = function(dlg, e){
32340             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32341                 var k = e.getKey();
32342                 if(keyCode instanceof Array){
32343                     for(var i = 0, len = keyCode.length; i < len; i++){
32344                         if(keyCode[i] == k){
32345                           fn.call(scope || window, dlg, k, e);
32346                           return;
32347                         }
32348                     }
32349                 }else{
32350                     if(k == keyCode){
32351                         fn.call(scope || window, dlg, k, e);
32352                     }
32353                 }
32354             }
32355         };
32356         this.on("keydown", handler);
32357         return this;
32358     },
32359
32360     /**
32361      * Returns the TabPanel component (creates it if it doesn't exist).
32362      * Note: If you wish to simply check for the existence of tabs without creating them,
32363      * check for a null 'tabs' property.
32364      * @return {Roo.TabPanel} The tabs component
32365      */
32366     getTabs : function(){
32367         if(!this.tabs){
32368             this.el.addClass("x-dlg-auto-tabs");
32369             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32370             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32371         }
32372         return this.tabs;
32373     },
32374
32375     /**
32376      * Adds a button to the footer section of the dialog.
32377      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32378      * object or a valid Roo.DomHelper element config
32379      * @param {Function} handler The function called when the button is clicked
32380      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32381      * @return {Roo.Button} The new button
32382      */
32383     addButton : function(config, handler, scope){
32384         var dh = Roo.DomHelper;
32385         if(!this.footer){
32386             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32387         }
32388         if(!this.btnContainer){
32389             var tb = this.footer.createChild({
32390
32391                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32392                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32393             }, null, true);
32394             this.btnContainer = tb.firstChild.firstChild.firstChild;
32395         }
32396         var bconfig = {
32397             handler: handler,
32398             scope: scope,
32399             minWidth: this.minButtonWidth,
32400             hideParent:true
32401         };
32402         if(typeof config == "string"){
32403             bconfig.text = config;
32404         }else{
32405             if(config.tag){
32406                 bconfig.dhconfig = config;
32407             }else{
32408                 Roo.apply(bconfig, config);
32409             }
32410         }
32411         var fc = false;
32412         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32413             bconfig.position = Math.max(0, bconfig.position);
32414             fc = this.btnContainer.childNodes[bconfig.position];
32415         }
32416          
32417         var btn = new Roo.Button(
32418             fc ? 
32419                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32420                 : this.btnContainer.appendChild(document.createElement("td")),
32421             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32422             bconfig
32423         );
32424         this.syncBodyHeight();
32425         if(!this.buttons){
32426             /**
32427              * Array of all the buttons that have been added to this dialog via addButton
32428              * @type Array
32429              */
32430             this.buttons = [];
32431         }
32432         this.buttons.push(btn);
32433         return btn;
32434     },
32435
32436     /**
32437      * Sets the default button to be focused when the dialog is displayed.
32438      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32439      * @return {Roo.BasicDialog} this
32440      */
32441     setDefaultButton : function(btn){
32442         this.defaultButton = btn;
32443         return this;
32444     },
32445
32446     // private
32447     getHeaderFooterHeight : function(safe){
32448         var height = 0;
32449         if(this.header){
32450            height += this.header.getHeight();
32451         }
32452         if(this.footer){
32453            var fm = this.footer.getMargins();
32454             height += (this.footer.getHeight()+fm.top+fm.bottom);
32455         }
32456         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32457         height += this.centerBg.getPadding("tb");
32458         return height;
32459     },
32460
32461     // private
32462     syncBodyHeight : function()
32463     {
32464         var bd = this.body, // the text
32465             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32466             bw = this.bwrap;
32467         var height = this.size.height - this.getHeaderFooterHeight(false);
32468         bd.setHeight(height-bd.getMargins("tb"));
32469         var hh = this.header.getHeight();
32470         var h = this.size.height-hh;
32471         cb.setHeight(h);
32472         
32473         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32474         bw.setHeight(h-cb.getPadding("tb"));
32475         
32476         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32477         bd.setWidth(bw.getWidth(true));
32478         if(this.tabs){
32479             this.tabs.syncHeight();
32480             if(Roo.isIE){
32481                 this.tabs.el.repaint();
32482             }
32483         }
32484     },
32485
32486     /**
32487      * Restores the previous state of the dialog if Roo.state is configured.
32488      * @return {Roo.BasicDialog} this
32489      */
32490     restoreState : function(){
32491         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32492         if(box && box.width){
32493             this.xy = [box.x, box.y];
32494             this.resizeTo(box.width, box.height);
32495         }
32496         return this;
32497     },
32498
32499     // private
32500     beforeShow : function(){
32501         this.expand();
32502         if(this.fixedcenter){
32503             this.xy = this.el.getCenterXY(true);
32504         }
32505         if(this.modal){
32506             Roo.get(document.body).addClass("x-body-masked");
32507             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32508             this.mask.show();
32509         }
32510         this.constrainXY();
32511     },
32512
32513     // private
32514     animShow : function(){
32515         var b = Roo.get(this.animateTarget).getBox();
32516         this.proxy.setSize(b.width, b.height);
32517         this.proxy.setLocation(b.x, b.y);
32518         this.proxy.show();
32519         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32520                     true, .35, this.showEl.createDelegate(this));
32521     },
32522
32523     /**
32524      * Shows the dialog.
32525      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32526      * @return {Roo.BasicDialog} this
32527      */
32528     show : function(animateTarget){
32529         if (this.fireEvent("beforeshow", this) === false){
32530             return;
32531         }
32532         if(this.syncHeightBeforeShow){
32533             this.syncBodyHeight();
32534         }else if(this.firstShow){
32535             this.firstShow = false;
32536             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32537         }
32538         this.animateTarget = animateTarget || this.animateTarget;
32539         if(!this.el.isVisible()){
32540             this.beforeShow();
32541             if(this.animateTarget && Roo.get(this.animateTarget)){
32542                 this.animShow();
32543             }else{
32544                 this.showEl();
32545             }
32546         }
32547         return this;
32548     },
32549
32550     // private
32551     showEl : function(){
32552         this.proxy.hide();
32553         this.el.setXY(this.xy);
32554         this.el.show();
32555         this.adjustAssets(true);
32556         this.toFront();
32557         this.focus();
32558         // IE peekaboo bug - fix found by Dave Fenwick
32559         if(Roo.isIE){
32560             this.el.repaint();
32561         }
32562         this.fireEvent("show", this);
32563     },
32564
32565     /**
32566      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32567      * dialog itself will receive focus.
32568      */
32569     focus : function(){
32570         if(this.defaultButton){
32571             this.defaultButton.focus();
32572         }else{
32573             this.focusEl.focus();
32574         }
32575     },
32576
32577     // private
32578     constrainXY : function(){
32579         if(this.constraintoviewport !== false){
32580             if(!this.viewSize){
32581                 if(this.container){
32582                     var s = this.container.getSize();
32583                     this.viewSize = [s.width, s.height];
32584                 }else{
32585                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32586                 }
32587             }
32588             var s = Roo.get(this.container||document).getScroll();
32589
32590             var x = this.xy[0], y = this.xy[1];
32591             var w = this.size.width, h = this.size.height;
32592             var vw = this.viewSize[0], vh = this.viewSize[1];
32593             // only move it if it needs it
32594             var moved = false;
32595             // first validate right/bottom
32596             if(x + w > vw+s.left){
32597                 x = vw - w;
32598                 moved = true;
32599             }
32600             if(y + h > vh+s.top){
32601                 y = vh - h;
32602                 moved = true;
32603             }
32604             // then make sure top/left isn't negative
32605             if(x < s.left){
32606                 x = s.left;
32607                 moved = true;
32608             }
32609             if(y < s.top){
32610                 y = s.top;
32611                 moved = true;
32612             }
32613             if(moved){
32614                 // cache xy
32615                 this.xy = [x, y];
32616                 if(this.isVisible()){
32617                     this.el.setLocation(x, y);
32618                     this.adjustAssets();
32619                 }
32620             }
32621         }
32622     },
32623
32624     // private
32625     onDrag : function(){
32626         if(!this.proxyDrag){
32627             this.xy = this.el.getXY();
32628             this.adjustAssets();
32629         }
32630     },
32631
32632     // private
32633     adjustAssets : function(doShow){
32634         var x = this.xy[0], y = this.xy[1];
32635         var w = this.size.width, h = this.size.height;
32636         if(doShow === true){
32637             if(this.shadow){
32638                 this.shadow.show(this.el);
32639             }
32640             if(this.shim){
32641                 this.shim.show();
32642             }
32643         }
32644         if(this.shadow && this.shadow.isVisible()){
32645             this.shadow.show(this.el);
32646         }
32647         if(this.shim && this.shim.isVisible()){
32648             this.shim.setBounds(x, y, w, h);
32649         }
32650     },
32651
32652     // private
32653     adjustViewport : function(w, h){
32654         if(!w || !h){
32655             w = Roo.lib.Dom.getViewWidth();
32656             h = Roo.lib.Dom.getViewHeight();
32657         }
32658         // cache the size
32659         this.viewSize = [w, h];
32660         if(this.modal && this.mask.isVisible()){
32661             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32662             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32663         }
32664         if(this.isVisible()){
32665             this.constrainXY();
32666         }
32667     },
32668
32669     /**
32670      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32671      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32672      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32673      */
32674     destroy : function(removeEl){
32675         if(this.isVisible()){
32676             this.animateTarget = null;
32677             this.hide();
32678         }
32679         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32680         if(this.tabs){
32681             this.tabs.destroy(removeEl);
32682         }
32683         Roo.destroy(
32684              this.shim,
32685              this.proxy,
32686              this.resizer,
32687              this.close,
32688              this.mask
32689         );
32690         if(this.dd){
32691             this.dd.unreg();
32692         }
32693         if(this.buttons){
32694            for(var i = 0, len = this.buttons.length; i < len; i++){
32695                this.buttons[i].destroy();
32696            }
32697         }
32698         this.el.removeAllListeners();
32699         if(removeEl === true){
32700             this.el.update("");
32701             this.el.remove();
32702         }
32703         Roo.DialogManager.unregister(this);
32704     },
32705
32706     // private
32707     startMove : function(){
32708         if(this.proxyDrag){
32709             this.proxy.show();
32710         }
32711         if(this.constraintoviewport !== false){
32712             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32713         }
32714     },
32715
32716     // private
32717     endMove : function(){
32718         if(!this.proxyDrag){
32719             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32720         }else{
32721             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32722             this.proxy.hide();
32723         }
32724         this.refreshSize();
32725         this.adjustAssets();
32726         this.focus();
32727         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32728     },
32729
32730     /**
32731      * Brings this dialog to the front of any other visible dialogs
32732      * @return {Roo.BasicDialog} this
32733      */
32734     toFront : function(){
32735         Roo.DialogManager.bringToFront(this);
32736         return this;
32737     },
32738
32739     /**
32740      * Sends this dialog to the back (under) of any other visible dialogs
32741      * @return {Roo.BasicDialog} this
32742      */
32743     toBack : function(){
32744         Roo.DialogManager.sendToBack(this);
32745         return this;
32746     },
32747
32748     /**
32749      * Centers this dialog in the viewport
32750      * @return {Roo.BasicDialog} this
32751      */
32752     center : function(){
32753         var xy = this.el.getCenterXY(true);
32754         this.moveTo(xy[0], xy[1]);
32755         return this;
32756     },
32757
32758     /**
32759      * Moves the dialog's top-left corner to the specified point
32760      * @param {Number} x
32761      * @param {Number} y
32762      * @return {Roo.BasicDialog} this
32763      */
32764     moveTo : function(x, y){
32765         this.xy = [x,y];
32766         if(this.isVisible()){
32767             this.el.setXY(this.xy);
32768             this.adjustAssets();
32769         }
32770         return this;
32771     },
32772
32773     /**
32774      * Aligns the dialog to the specified element
32775      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32776      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32777      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32778      * @return {Roo.BasicDialog} this
32779      */
32780     alignTo : function(element, position, offsets){
32781         this.xy = this.el.getAlignToXY(element, position, offsets);
32782         if(this.isVisible()){
32783             this.el.setXY(this.xy);
32784             this.adjustAssets();
32785         }
32786         return this;
32787     },
32788
32789     /**
32790      * Anchors an element to another element and realigns it when the window is resized.
32791      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32792      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32793      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32794      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32795      * is a number, it is used as the buffer delay (defaults to 50ms).
32796      * @return {Roo.BasicDialog} this
32797      */
32798     anchorTo : function(el, alignment, offsets, monitorScroll){
32799         var action = function(){
32800             this.alignTo(el, alignment, offsets);
32801         };
32802         Roo.EventManager.onWindowResize(action, this);
32803         var tm = typeof monitorScroll;
32804         if(tm != 'undefined'){
32805             Roo.EventManager.on(window, 'scroll', action, this,
32806                 {buffer: tm == 'number' ? monitorScroll : 50});
32807         }
32808         action.call(this);
32809         return this;
32810     },
32811
32812     /**
32813      * Returns true if the dialog is visible
32814      * @return {Boolean}
32815      */
32816     isVisible : function(){
32817         return this.el.isVisible();
32818     },
32819
32820     // private
32821     animHide : function(callback){
32822         var b = Roo.get(this.animateTarget).getBox();
32823         this.proxy.show();
32824         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32825         this.el.hide();
32826         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32827                     this.hideEl.createDelegate(this, [callback]));
32828     },
32829
32830     /**
32831      * Hides the dialog.
32832      * @param {Function} callback (optional) Function to call when the dialog is hidden
32833      * @return {Roo.BasicDialog} this
32834      */
32835     hide : function(callback){
32836         if (this.fireEvent("beforehide", this) === false){
32837             return;
32838         }
32839         if(this.shadow){
32840             this.shadow.hide();
32841         }
32842         if(this.shim) {
32843           this.shim.hide();
32844         }
32845         // sometimes animateTarget seems to get set.. causing problems...
32846         // this just double checks..
32847         if(this.animateTarget && Roo.get(this.animateTarget)) {
32848            this.animHide(callback);
32849         }else{
32850             this.el.hide();
32851             this.hideEl(callback);
32852         }
32853         return this;
32854     },
32855
32856     // private
32857     hideEl : function(callback){
32858         this.proxy.hide();
32859         if(this.modal){
32860             this.mask.hide();
32861             Roo.get(document.body).removeClass("x-body-masked");
32862         }
32863         this.fireEvent("hide", this);
32864         if(typeof callback == "function"){
32865             callback();
32866         }
32867     },
32868
32869     // private
32870     hideAction : function(){
32871         this.setLeft("-10000px");
32872         this.setTop("-10000px");
32873         this.setStyle("visibility", "hidden");
32874     },
32875
32876     // private
32877     refreshSize : function(){
32878         this.size = this.el.getSize();
32879         this.xy = this.el.getXY();
32880         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32881     },
32882
32883     // private
32884     // z-index is managed by the DialogManager and may be overwritten at any time
32885     setZIndex : function(index){
32886         if(this.modal){
32887             this.mask.setStyle("z-index", index);
32888         }
32889         if(this.shim){
32890             this.shim.setStyle("z-index", ++index);
32891         }
32892         if(this.shadow){
32893             this.shadow.setZIndex(++index);
32894         }
32895         this.el.setStyle("z-index", ++index);
32896         if(this.proxy){
32897             this.proxy.setStyle("z-index", ++index);
32898         }
32899         if(this.resizer){
32900             this.resizer.proxy.setStyle("z-index", ++index);
32901         }
32902
32903         this.lastZIndex = index;
32904     },
32905
32906     /**
32907      * Returns the element for this dialog
32908      * @return {Roo.Element} The underlying dialog Element
32909      */
32910     getEl : function(){
32911         return this.el;
32912     }
32913 });
32914
32915 /**
32916  * @class Roo.DialogManager
32917  * Provides global access to BasicDialogs that have been created and
32918  * support for z-indexing (layering) multiple open dialogs.
32919  */
32920 Roo.DialogManager = function(){
32921     var list = {};
32922     var accessList = [];
32923     var front = null;
32924
32925     // private
32926     var sortDialogs = function(d1, d2){
32927         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32928     };
32929
32930     // private
32931     var orderDialogs = function(){
32932         accessList.sort(sortDialogs);
32933         var seed = Roo.DialogManager.zseed;
32934         for(var i = 0, len = accessList.length; i < len; i++){
32935             var dlg = accessList[i];
32936             if(dlg){
32937                 dlg.setZIndex(seed + (i*10));
32938             }
32939         }
32940     };
32941
32942     return {
32943         /**
32944          * The starting z-index for BasicDialogs (defaults to 9000)
32945          * @type Number The z-index value
32946          */
32947         zseed : 9000,
32948
32949         // private
32950         register : function(dlg){
32951             list[dlg.id] = dlg;
32952             accessList.push(dlg);
32953         },
32954
32955         // private
32956         unregister : function(dlg){
32957             delete list[dlg.id];
32958             var i=0;
32959             var len=0;
32960             if(!accessList.indexOf){
32961                 for(  i = 0, len = accessList.length; i < len; i++){
32962                     if(accessList[i] == dlg){
32963                         accessList.splice(i, 1);
32964                         return;
32965                     }
32966                 }
32967             }else{
32968                  i = accessList.indexOf(dlg);
32969                 if(i != -1){
32970                     accessList.splice(i, 1);
32971                 }
32972             }
32973         },
32974
32975         /**
32976          * Gets a registered dialog by id
32977          * @param {String/Object} id The id of the dialog or a dialog
32978          * @return {Roo.BasicDialog} this
32979          */
32980         get : function(id){
32981             return typeof id == "object" ? id : list[id];
32982         },
32983
32984         /**
32985          * Brings the specified dialog to the front
32986          * @param {String/Object} dlg The id of the dialog or a dialog
32987          * @return {Roo.BasicDialog} this
32988          */
32989         bringToFront : function(dlg){
32990             dlg = this.get(dlg);
32991             if(dlg != front){
32992                 front = dlg;
32993                 dlg._lastAccess = new Date().getTime();
32994                 orderDialogs();
32995             }
32996             return dlg;
32997         },
32998
32999         /**
33000          * Sends the specified dialog to the back
33001          * @param {String/Object} dlg The id of the dialog or a dialog
33002          * @return {Roo.BasicDialog} this
33003          */
33004         sendToBack : function(dlg){
33005             dlg = this.get(dlg);
33006             dlg._lastAccess = -(new Date().getTime());
33007             orderDialogs();
33008             return dlg;
33009         },
33010
33011         /**
33012          * Hides all dialogs
33013          */
33014         hideAll : function(){
33015             for(var id in list){
33016                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
33017                     list[id].hide();
33018                 }
33019             }
33020         }
33021     };
33022 }();
33023
33024 /**
33025  * @class Roo.LayoutDialog
33026  * @extends Roo.BasicDialog
33027  * Dialog which provides adjustments for working with a layout in a Dialog.
33028  * Add your necessary layout config options to the dialog's config.<br>
33029  * Example usage (including a nested layout):
33030  * <pre><code>
33031 if(!dialog){
33032     dialog = new Roo.LayoutDialog("download-dlg", {
33033         modal: true,
33034         width:600,
33035         height:450,
33036         shadow:true,
33037         minWidth:500,
33038         minHeight:350,
33039         autoTabs:true,
33040         proxyDrag:true,
33041         // layout config merges with the dialog config
33042         center:{
33043             tabPosition: "top",
33044             alwaysShowTabs: true
33045         }
33046     });
33047     dialog.addKeyListener(27, dialog.hide, dialog);
33048     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33049     dialog.addButton("Build It!", this.getDownload, this);
33050
33051     // we can even add nested layouts
33052     var innerLayout = new Roo.BorderLayout("dl-inner", {
33053         east: {
33054             initialSize: 200,
33055             autoScroll:true,
33056             split:true
33057         },
33058         center: {
33059             autoScroll:true
33060         }
33061     });
33062     innerLayout.beginUpdate();
33063     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33064     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33065     innerLayout.endUpdate(true);
33066
33067     var layout = dialog.getLayout();
33068     layout.beginUpdate();
33069     layout.add("center", new Roo.ContentPanel("standard-panel",
33070                         {title: "Download the Source", fitToFrame:true}));
33071     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33072                {title: "Build your own roo.js"}));
33073     layout.getRegion("center").showPanel(sp);
33074     layout.endUpdate();
33075 }
33076 </code></pre>
33077     * @constructor
33078     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33079     * @param {Object} config configuration options
33080   */
33081 Roo.LayoutDialog = function(el, cfg){
33082     
33083     var config=  cfg;
33084     if (typeof(cfg) == 'undefined') {
33085         config = Roo.apply({}, el);
33086         // not sure why we use documentElement here.. - it should always be body.
33087         // IE7 borks horribly if we use documentElement.
33088         // webkit also does not like documentElement - it creates a body element...
33089         el = Roo.get( document.body || document.documentElement ).createChild();
33090         //config.autoCreate = true;
33091     }
33092     
33093     
33094     config.autoTabs = false;
33095     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33096     this.body.setStyle({overflow:"hidden", position:"relative"});
33097     this.layout = new Roo.BorderLayout(this.body.dom, config);
33098     this.layout.monitorWindowResize = false;
33099     this.el.addClass("x-dlg-auto-layout");
33100     // fix case when center region overwrites center function
33101     this.center = Roo.BasicDialog.prototype.center;
33102     this.on("show", this.layout.layout, this.layout, true);
33103     if (config.items) {
33104         var xitems = config.items;
33105         delete config.items;
33106         Roo.each(xitems, this.addxtype, this);
33107     }
33108     
33109     
33110 };
33111 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33112     /**
33113      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33114      * @deprecated
33115      */
33116     endUpdate : function(){
33117         this.layout.endUpdate();
33118     },
33119
33120     /**
33121      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33122      *  @deprecated
33123      */
33124     beginUpdate : function(){
33125         this.layout.beginUpdate();
33126     },
33127
33128     /**
33129      * Get the BorderLayout for this dialog
33130      * @return {Roo.BorderLayout}
33131      */
33132     getLayout : function(){
33133         return this.layout;
33134     },
33135
33136     showEl : function(){
33137         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33138         if(Roo.isIE7){
33139             this.layout.layout();
33140         }
33141     },
33142
33143     // private
33144     // Use the syncHeightBeforeShow config option to control this automatically
33145     syncBodyHeight : function(){
33146         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33147         if(this.layout){this.layout.layout();}
33148     },
33149     
33150       /**
33151      * Add an xtype element (actually adds to the layout.)
33152      * @return {Object} xdata xtype object data.
33153      */
33154     
33155     addxtype : function(c) {
33156         return this.layout.addxtype(c);
33157     }
33158 });/*
33159  * Based on:
33160  * Ext JS Library 1.1.1
33161  * Copyright(c) 2006-2007, Ext JS, LLC.
33162  *
33163  * Originally Released Under LGPL - original licence link has changed is not relivant.
33164  *
33165  * Fork - LGPL
33166  * <script type="text/javascript">
33167  */
33168  
33169 /**
33170  * @class Roo.MessageBox
33171  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33172  * Example usage:
33173  *<pre><code>
33174 // Basic alert:
33175 Roo.Msg.alert('Status', 'Changes saved successfully.');
33176
33177 // Prompt for user data:
33178 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33179     if (btn == 'ok'){
33180         // process text value...
33181     }
33182 });
33183
33184 // Show a dialog using config options:
33185 Roo.Msg.show({
33186    title:'Save Changes?',
33187    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33188    buttons: Roo.Msg.YESNOCANCEL,
33189    fn: processResult,
33190    animEl: 'elId'
33191 });
33192 </code></pre>
33193  * @singleton
33194  */
33195 Roo.MessageBox = function(){
33196     var dlg, opt, mask, waitTimer;
33197     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33198     var buttons, activeTextEl, bwidth;
33199
33200     // private
33201     var handleButton = function(button){
33202         dlg.hide();
33203         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33204     };
33205
33206     // private
33207     var handleHide = function(){
33208         if(opt && opt.cls){
33209             dlg.el.removeClass(opt.cls);
33210         }
33211         if(waitTimer){
33212             Roo.TaskMgr.stop(waitTimer);
33213             waitTimer = null;
33214         }
33215     };
33216
33217     // private
33218     var updateButtons = function(b){
33219         var width = 0;
33220         if(!b){
33221             buttons["ok"].hide();
33222             buttons["cancel"].hide();
33223             buttons["yes"].hide();
33224             buttons["no"].hide();
33225             dlg.footer.dom.style.display = 'none';
33226             return width;
33227         }
33228         dlg.footer.dom.style.display = '';
33229         for(var k in buttons){
33230             if(typeof buttons[k] != "function"){
33231                 if(b[k]){
33232                     buttons[k].show();
33233                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33234                     width += buttons[k].el.getWidth()+15;
33235                 }else{
33236                     buttons[k].hide();
33237                 }
33238             }
33239         }
33240         return width;
33241     };
33242
33243     // private
33244     var handleEsc = function(d, k, e){
33245         if(opt && opt.closable !== false){
33246             dlg.hide();
33247         }
33248         if(e){
33249             e.stopEvent();
33250         }
33251     };
33252
33253     return {
33254         /**
33255          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33256          * @return {Roo.BasicDialog} The BasicDialog element
33257          */
33258         getDialog : function(){
33259            if(!dlg){
33260                 dlg = new Roo.BasicDialog("x-msg-box", {
33261                     autoCreate : true,
33262                     shadow: true,
33263                     draggable: true,
33264                     resizable:false,
33265                     constraintoviewport:false,
33266                     fixedcenter:true,
33267                     collapsible : false,
33268                     shim:true,
33269                     modal: true,
33270                     width:400, height:100,
33271                     buttonAlign:"center",
33272                     closeClick : function(){
33273                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33274                             handleButton("no");
33275                         }else{
33276                             handleButton("cancel");
33277                         }
33278                     }
33279                 });
33280                 dlg.on("hide", handleHide);
33281                 mask = dlg.mask;
33282                 dlg.addKeyListener(27, handleEsc);
33283                 buttons = {};
33284                 var bt = this.buttonText;
33285                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33286                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33287                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33288                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33289                 bodyEl = dlg.body.createChild({
33290
33291                     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>'
33292                 });
33293                 msgEl = bodyEl.dom.firstChild;
33294                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33295                 textboxEl.enableDisplayMode();
33296                 textboxEl.addKeyListener([10,13], function(){
33297                     if(dlg.isVisible() && opt && opt.buttons){
33298                         if(opt.buttons.ok){
33299                             handleButton("ok");
33300                         }else if(opt.buttons.yes){
33301                             handleButton("yes");
33302                         }
33303                     }
33304                 });
33305                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33306                 textareaEl.enableDisplayMode();
33307                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33308                 progressEl.enableDisplayMode();
33309                 var pf = progressEl.dom.firstChild;
33310                 if (pf) {
33311                     pp = Roo.get(pf.firstChild);
33312                     pp.setHeight(pf.offsetHeight);
33313                 }
33314                 
33315             }
33316             return dlg;
33317         },
33318
33319         /**
33320          * Updates the message box body text
33321          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33322          * the XHTML-compliant non-breaking space character '&amp;#160;')
33323          * @return {Roo.MessageBox} This message box
33324          */
33325         updateText : function(text){
33326             if(!dlg.isVisible() && !opt.width){
33327                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33328             }
33329             msgEl.innerHTML = text || '&#160;';
33330       
33331             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33332             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33333             var w = Math.max(
33334                     Math.min(opt.width || cw , this.maxWidth), 
33335                     Math.max(opt.minWidth || this.minWidth, bwidth)
33336             );
33337             if(opt.prompt){
33338                 activeTextEl.setWidth(w);
33339             }
33340             if(dlg.isVisible()){
33341                 dlg.fixedcenter = false;
33342             }
33343             // to big, make it scroll. = But as usual stupid IE does not support
33344             // !important..
33345             
33346             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33347                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33348                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33349             } else {
33350                 bodyEl.dom.style.height = '';
33351                 bodyEl.dom.style.overflowY = '';
33352             }
33353             if (cw > w) {
33354                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33355             } else {
33356                 bodyEl.dom.style.overflowX = '';
33357             }
33358             
33359             dlg.setContentSize(w, bodyEl.getHeight());
33360             if(dlg.isVisible()){
33361                 dlg.fixedcenter = true;
33362             }
33363             return this;
33364         },
33365
33366         /**
33367          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33368          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33369          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33370          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33371          * @return {Roo.MessageBox} This message box
33372          */
33373         updateProgress : function(value, text){
33374             if(text){
33375                 this.updateText(text);
33376             }
33377             if (pp) { // weird bug on my firefox - for some reason this is not defined
33378                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33379             }
33380             return this;
33381         },        
33382
33383         /**
33384          * Returns true if the message box is currently displayed
33385          * @return {Boolean} True if the message box is visible, else false
33386          */
33387         isVisible : function(){
33388             return dlg && dlg.isVisible();  
33389         },
33390
33391         /**
33392          * Hides the message box if it is displayed
33393          */
33394         hide : function(){
33395             if(this.isVisible()){
33396                 dlg.hide();
33397             }  
33398         },
33399
33400         /**
33401          * Displays a new message box, or reinitializes an existing message box, based on the config options
33402          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33403          * The following config object properties are supported:
33404          * <pre>
33405 Property    Type             Description
33406 ----------  ---------------  ------------------------------------------------------------------------------------
33407 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33408                                    closes (defaults to undefined)
33409 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33410                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33411 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33412                                    progress and wait dialogs will ignore this property and always hide the
33413                                    close button as they can only be closed programmatically.
33414 cls               String           A custom CSS class to apply to the message box element
33415 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33416                                    displayed (defaults to 75)
33417 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33418                                    function will be btn (the name of the button that was clicked, if applicable,
33419                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33420                                    Progress and wait dialogs will ignore this option since they do not respond to
33421                                    user actions and can only be closed programmatically, so any required function
33422                                    should be called by the same code after it closes the dialog.
33423 icon              String           A CSS class that provides a background image to be used as an icon for
33424                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33425 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33426 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33427 modal             Boolean          False to allow user interaction with the page while the message box is
33428                                    displayed (defaults to true)
33429 msg               String           A string that will replace the existing message box body text (defaults
33430                                    to the XHTML-compliant non-breaking space character '&#160;')
33431 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33432 progress          Boolean          True to display a progress bar (defaults to false)
33433 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33434 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33435 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33436 title             String           The title text
33437 value             String           The string value to set into the active textbox element if displayed
33438 wait              Boolean          True to display a progress bar (defaults to false)
33439 width             Number           The width of the dialog in pixels
33440 </pre>
33441          *
33442          * Example usage:
33443          * <pre><code>
33444 Roo.Msg.show({
33445    title: 'Address',
33446    msg: 'Please enter your address:',
33447    width: 300,
33448    buttons: Roo.MessageBox.OKCANCEL,
33449    multiline: true,
33450    fn: saveAddress,
33451    animEl: 'addAddressBtn'
33452 });
33453 </code></pre>
33454          * @param {Object} config Configuration options
33455          * @return {Roo.MessageBox} This message box
33456          */
33457         show : function(options)
33458         {
33459             
33460             // this causes nightmares if you show one dialog after another
33461             // especially on callbacks..
33462              
33463             if(this.isVisible()){
33464                 
33465                 this.hide();
33466                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33467                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33468                 Roo.log("New Dialog Message:" +  options.msg )
33469                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33470                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33471                 
33472             }
33473             var d = this.getDialog();
33474             opt = options;
33475             d.setTitle(opt.title || "&#160;");
33476             d.close.setDisplayed(opt.closable !== false);
33477             activeTextEl = textboxEl;
33478             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33479             if(opt.prompt){
33480                 if(opt.multiline){
33481                     textboxEl.hide();
33482                     textareaEl.show();
33483                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33484                         opt.multiline : this.defaultTextHeight);
33485                     activeTextEl = textareaEl;
33486                 }else{
33487                     textboxEl.show();
33488                     textareaEl.hide();
33489                 }
33490             }else{
33491                 textboxEl.hide();
33492                 textareaEl.hide();
33493             }
33494             progressEl.setDisplayed(opt.progress === true);
33495             this.updateProgress(0);
33496             activeTextEl.dom.value = opt.value || "";
33497             if(opt.prompt){
33498                 dlg.setDefaultButton(activeTextEl);
33499             }else{
33500                 var bs = opt.buttons;
33501                 var db = null;
33502                 if(bs && bs.ok){
33503                     db = buttons["ok"];
33504                 }else if(bs && bs.yes){
33505                     db = buttons["yes"];
33506                 }
33507                 dlg.setDefaultButton(db);
33508             }
33509             bwidth = updateButtons(opt.buttons);
33510             this.updateText(opt.msg);
33511             if(opt.cls){
33512                 d.el.addClass(opt.cls);
33513             }
33514             d.proxyDrag = opt.proxyDrag === true;
33515             d.modal = opt.modal !== false;
33516             d.mask = opt.modal !== false ? mask : false;
33517             if(!d.isVisible()){
33518                 // force it to the end of the z-index stack so it gets a cursor in FF
33519                 document.body.appendChild(dlg.el.dom);
33520                 d.animateTarget = null;
33521                 d.show(options.animEl);
33522             }
33523             return this;
33524         },
33525
33526         /**
33527          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33528          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33529          * and closing the message box when the process is complete.
33530          * @param {String} title The title bar text
33531          * @param {String} msg The message box body text
33532          * @return {Roo.MessageBox} This message box
33533          */
33534         progress : function(title, msg){
33535             this.show({
33536                 title : title,
33537                 msg : msg,
33538                 buttons: false,
33539                 progress:true,
33540                 closable:false,
33541                 minWidth: this.minProgressWidth,
33542                 modal : true
33543             });
33544             return this;
33545         },
33546
33547         /**
33548          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33549          * If a callback function is passed it will be called after the user clicks the button, and the
33550          * id of the button that was clicked will be passed as the only parameter to the callback
33551          * (could also be the top-right close button).
33552          * @param {String} title The title bar text
33553          * @param {String} msg The message box body text
33554          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33555          * @param {Object} scope (optional) The scope of the callback function
33556          * @return {Roo.MessageBox} This message box
33557          */
33558         alert : function(title, msg, fn, scope){
33559             this.show({
33560                 title : title,
33561                 msg : msg,
33562                 buttons: this.OK,
33563                 fn: fn,
33564                 scope : scope,
33565                 modal : true
33566             });
33567             return this;
33568         },
33569
33570         /**
33571          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33572          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33573          * You are responsible for closing the message box when the process is complete.
33574          * @param {String} msg The message box body text
33575          * @param {String} title (optional) The title bar text
33576          * @return {Roo.MessageBox} This message box
33577          */
33578         wait : function(msg, title){
33579             this.show({
33580                 title : title,
33581                 msg : msg,
33582                 buttons: false,
33583                 closable:false,
33584                 progress:true,
33585                 modal:true,
33586                 width:300,
33587                 wait:true
33588             });
33589             waitTimer = Roo.TaskMgr.start({
33590                 run: function(i){
33591                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33592                 },
33593                 interval: 1000
33594             });
33595             return this;
33596         },
33597
33598         /**
33599          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33600          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33601          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33602          * @param {String} title The title bar text
33603          * @param {String} msg The message box body text
33604          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33605          * @param {Object} scope (optional) The scope of the callback function
33606          * @return {Roo.MessageBox} This message box
33607          */
33608         confirm : function(title, msg, fn, scope){
33609             this.show({
33610                 title : title,
33611                 msg : msg,
33612                 buttons: this.YESNO,
33613                 fn: fn,
33614                 scope : scope,
33615                 modal : true
33616             });
33617             return this;
33618         },
33619
33620         /**
33621          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33622          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33623          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33624          * (could also be the top-right close button) and the text that was entered will be passed as the two
33625          * parameters to the callback.
33626          * @param {String} title The title bar text
33627          * @param {String} msg The message box body text
33628          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33629          * @param {Object} scope (optional) The scope of the callback function
33630          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33631          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33632          * @return {Roo.MessageBox} This message box
33633          */
33634         prompt : function(title, msg, fn, scope, multiline){
33635             this.show({
33636                 title : title,
33637                 msg : msg,
33638                 buttons: this.OKCANCEL,
33639                 fn: fn,
33640                 minWidth:250,
33641                 scope : scope,
33642                 prompt:true,
33643                 multiline: multiline,
33644                 modal : true
33645             });
33646             return this;
33647         },
33648
33649         /**
33650          * Button config that displays a single OK button
33651          * @type Object
33652          */
33653         OK : {ok:true},
33654         /**
33655          * Button config that displays Yes and No buttons
33656          * @type Object
33657          */
33658         YESNO : {yes:true, no:true},
33659         /**
33660          * Button config that displays OK and Cancel buttons
33661          * @type Object
33662          */
33663         OKCANCEL : {ok:true, cancel:true},
33664         /**
33665          * Button config that displays Yes, No and Cancel buttons
33666          * @type Object
33667          */
33668         YESNOCANCEL : {yes:true, no:true, cancel:true},
33669
33670         /**
33671          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33672          * @type Number
33673          */
33674         defaultTextHeight : 75,
33675         /**
33676          * The maximum width in pixels of the message box (defaults to 600)
33677          * @type Number
33678          */
33679         maxWidth : 600,
33680         /**
33681          * The minimum width in pixels of the message box (defaults to 100)
33682          * @type Number
33683          */
33684         minWidth : 100,
33685         /**
33686          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33687          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33688          * @type Number
33689          */
33690         minProgressWidth : 250,
33691         /**
33692          * An object containing the default button text strings that can be overriden for localized language support.
33693          * Supported properties are: ok, cancel, yes and no.
33694          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33695          * @type Object
33696          */
33697         buttonText : {
33698             ok : "OK",
33699             cancel : "Cancel",
33700             yes : "Yes",
33701             no : "No"
33702         }
33703     };
33704 }();
33705
33706 /**
33707  * Shorthand for {@link Roo.MessageBox}
33708  */
33709 Roo.Msg = Roo.MessageBox;/*
33710  * Based on:
33711  * Ext JS Library 1.1.1
33712  * Copyright(c) 2006-2007, Ext JS, LLC.
33713  *
33714  * Originally Released Under LGPL - original licence link has changed is not relivant.
33715  *
33716  * Fork - LGPL
33717  * <script type="text/javascript">
33718  */
33719 /**
33720  * @class Roo.QuickTips
33721  * Provides attractive and customizable tooltips for any element.
33722  * @singleton
33723  */
33724 Roo.QuickTips = function(){
33725     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33726     var ce, bd, xy, dd;
33727     var visible = false, disabled = true, inited = false;
33728     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33729     
33730     var onOver = function(e){
33731         if(disabled){
33732             return;
33733         }
33734         var t = e.getTarget();
33735         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33736             return;
33737         }
33738         if(ce && t == ce.el){
33739             clearTimeout(hideProc);
33740             return;
33741         }
33742         if(t && tagEls[t.id]){
33743             tagEls[t.id].el = t;
33744             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33745             return;
33746         }
33747         var ttp, et = Roo.fly(t);
33748         var ns = cfg.namespace;
33749         if(tm.interceptTitles && t.title){
33750             ttp = t.title;
33751             t.qtip = ttp;
33752             t.removeAttribute("title");
33753             e.preventDefault();
33754         }else{
33755             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33756         }
33757         if(ttp){
33758             showProc = show.defer(tm.showDelay, tm, [{
33759                 el: t, 
33760                 text: ttp.replace(/\\n/g,'<br/>'),
33761                 width: et.getAttributeNS(ns, cfg.width),
33762                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33763                 title: et.getAttributeNS(ns, cfg.title),
33764                     cls: et.getAttributeNS(ns, cfg.cls)
33765             }]);
33766         }
33767     };
33768     
33769     var onOut = function(e){
33770         clearTimeout(showProc);
33771         var t = e.getTarget();
33772         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33773             hideProc = setTimeout(hide, tm.hideDelay);
33774         }
33775     };
33776     
33777     var onMove = function(e){
33778         if(disabled){
33779             return;
33780         }
33781         xy = e.getXY();
33782         xy[1] += 18;
33783         if(tm.trackMouse && ce){
33784             el.setXY(xy);
33785         }
33786     };
33787     
33788     var onDown = function(e){
33789         clearTimeout(showProc);
33790         clearTimeout(hideProc);
33791         if(!e.within(el)){
33792             if(tm.hideOnClick){
33793                 hide();
33794                 tm.disable();
33795                 tm.enable.defer(100, tm);
33796             }
33797         }
33798     };
33799     
33800     var getPad = function(){
33801         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33802     };
33803
33804     var show = function(o){
33805         if(disabled){
33806             return;
33807         }
33808         clearTimeout(dismissProc);
33809         ce = o;
33810         if(removeCls){ // in case manually hidden
33811             el.removeClass(removeCls);
33812             removeCls = null;
33813         }
33814         if(ce.cls){
33815             el.addClass(ce.cls);
33816             removeCls = ce.cls;
33817         }
33818         if(ce.title){
33819             tipTitle.update(ce.title);
33820             tipTitle.show();
33821         }else{
33822             tipTitle.update('');
33823             tipTitle.hide();
33824         }
33825         el.dom.style.width  = tm.maxWidth+'px';
33826         //tipBody.dom.style.width = '';
33827         tipBodyText.update(o.text);
33828         var p = getPad(), w = ce.width;
33829         if(!w){
33830             var td = tipBodyText.dom;
33831             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33832             if(aw > tm.maxWidth){
33833                 w = tm.maxWidth;
33834             }else if(aw < tm.minWidth){
33835                 w = tm.minWidth;
33836             }else{
33837                 w = aw;
33838             }
33839         }
33840         //tipBody.setWidth(w);
33841         el.setWidth(parseInt(w, 10) + p);
33842         if(ce.autoHide === false){
33843             close.setDisplayed(true);
33844             if(dd){
33845                 dd.unlock();
33846             }
33847         }else{
33848             close.setDisplayed(false);
33849             if(dd){
33850                 dd.lock();
33851             }
33852         }
33853         if(xy){
33854             el.avoidY = xy[1]-18;
33855             el.setXY(xy);
33856         }
33857         if(tm.animate){
33858             el.setOpacity(.1);
33859             el.setStyle("visibility", "visible");
33860             el.fadeIn({callback: afterShow});
33861         }else{
33862             afterShow();
33863         }
33864     };
33865     
33866     var afterShow = function(){
33867         if(ce){
33868             el.show();
33869             esc.enable();
33870             if(tm.autoDismiss && ce.autoHide !== false){
33871                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33872             }
33873         }
33874     };
33875     
33876     var hide = function(noanim){
33877         clearTimeout(dismissProc);
33878         clearTimeout(hideProc);
33879         ce = null;
33880         if(el.isVisible()){
33881             esc.disable();
33882             if(noanim !== true && tm.animate){
33883                 el.fadeOut({callback: afterHide});
33884             }else{
33885                 afterHide();
33886             } 
33887         }
33888     };
33889     
33890     var afterHide = function(){
33891         el.hide();
33892         if(removeCls){
33893             el.removeClass(removeCls);
33894             removeCls = null;
33895         }
33896     };
33897     
33898     return {
33899         /**
33900         * @cfg {Number} minWidth
33901         * The minimum width of the quick tip (defaults to 40)
33902         */
33903        minWidth : 40,
33904         /**
33905         * @cfg {Number} maxWidth
33906         * The maximum width of the quick tip (defaults to 300)
33907         */
33908        maxWidth : 300,
33909         /**
33910         * @cfg {Boolean} interceptTitles
33911         * True to automatically use the element's DOM title value if available (defaults to false)
33912         */
33913        interceptTitles : false,
33914         /**
33915         * @cfg {Boolean} trackMouse
33916         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33917         */
33918        trackMouse : false,
33919         /**
33920         * @cfg {Boolean} hideOnClick
33921         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33922         */
33923        hideOnClick : true,
33924         /**
33925         * @cfg {Number} showDelay
33926         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33927         */
33928        showDelay : 500,
33929         /**
33930         * @cfg {Number} hideDelay
33931         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33932         */
33933        hideDelay : 200,
33934         /**
33935         * @cfg {Boolean} autoHide
33936         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33937         * Used in conjunction with hideDelay.
33938         */
33939        autoHide : true,
33940         /**
33941         * @cfg {Boolean}
33942         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33943         * (defaults to true).  Used in conjunction with autoDismissDelay.
33944         */
33945        autoDismiss : true,
33946         /**
33947         * @cfg {Number}
33948         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33949         */
33950        autoDismissDelay : 5000,
33951        /**
33952         * @cfg {Boolean} animate
33953         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33954         */
33955        animate : false,
33956
33957        /**
33958         * @cfg {String} title
33959         * Title text to display (defaults to '').  This can be any valid HTML markup.
33960         */
33961         title: '',
33962        /**
33963         * @cfg {String} text
33964         * Body text to display (defaults to '').  This can be any valid HTML markup.
33965         */
33966         text : '',
33967        /**
33968         * @cfg {String} cls
33969         * A CSS class to apply to the base quick tip element (defaults to '').
33970         */
33971         cls : '',
33972        /**
33973         * @cfg {Number} width
33974         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33975         * minWidth or maxWidth.
33976         */
33977         width : null,
33978
33979     /**
33980      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33981      * or display QuickTips in a page.
33982      */
33983        init : function(){
33984           tm = Roo.QuickTips;
33985           cfg = tm.tagConfig;
33986           if(!inited){
33987               if(!Roo.isReady){ // allow calling of init() before onReady
33988                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33989                   return;
33990               }
33991               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33992               el.fxDefaults = {stopFx: true};
33993               // maximum custom styling
33994               //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>');
33995               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>');              
33996               tipTitle = el.child('h3');
33997               tipTitle.enableDisplayMode("block");
33998               tipBody = el.child('div.x-tip-bd');
33999               tipBodyText = el.child('div.x-tip-bd-inner');
34000               //bdLeft = el.child('div.x-tip-bd-left');
34001               //bdRight = el.child('div.x-tip-bd-right');
34002               close = el.child('div.x-tip-close');
34003               close.enableDisplayMode("block");
34004               close.on("click", hide);
34005               var d = Roo.get(document);
34006               d.on("mousedown", onDown);
34007               d.on("mouseover", onOver);
34008               d.on("mouseout", onOut);
34009               d.on("mousemove", onMove);
34010               esc = d.addKeyListener(27, hide);
34011               esc.disable();
34012               if(Roo.dd.DD){
34013                   dd = el.initDD("default", null, {
34014                       onDrag : function(){
34015                           el.sync();  
34016                       }
34017                   });
34018                   dd.setHandleElId(tipTitle.id);
34019                   dd.lock();
34020               }
34021               inited = true;
34022           }
34023           this.enable(); 
34024        },
34025
34026     /**
34027      * Configures a new quick tip instance and assigns it to a target element.  The following config options
34028      * are supported:
34029      * <pre>
34030 Property    Type                   Description
34031 ----------  ---------------------  ------------------------------------------------------------------------
34032 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
34033      * </ul>
34034      * @param {Object} config The config object
34035      */
34036        register : function(config){
34037            var cs = config instanceof Array ? config : arguments;
34038            for(var i = 0, len = cs.length; i < len; i++) {
34039                var c = cs[i];
34040                var target = c.target;
34041                if(target){
34042                    if(target instanceof Array){
34043                        for(var j = 0, jlen = target.length; j < jlen; j++){
34044                            tagEls[target[j]] = c;
34045                        }
34046                    }else{
34047                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34048                    }
34049                }
34050            }
34051        },
34052
34053     /**
34054      * Removes this quick tip from its element and destroys it.
34055      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34056      */
34057        unregister : function(el){
34058            delete tagEls[Roo.id(el)];
34059        },
34060
34061     /**
34062      * Enable this quick tip.
34063      */
34064        enable : function(){
34065            if(inited && disabled){
34066                locks.pop();
34067                if(locks.length < 1){
34068                    disabled = false;
34069                }
34070            }
34071        },
34072
34073     /**
34074      * Disable this quick tip.
34075      */
34076        disable : function(){
34077           disabled = true;
34078           clearTimeout(showProc);
34079           clearTimeout(hideProc);
34080           clearTimeout(dismissProc);
34081           if(ce){
34082               hide(true);
34083           }
34084           locks.push(1);
34085        },
34086
34087     /**
34088      * Returns true if the quick tip is enabled, else false.
34089      */
34090        isEnabled : function(){
34091             return !disabled;
34092        },
34093
34094         // private
34095        tagConfig : {
34096            namespace : "roo", // was ext?? this may break..
34097            alt_namespace : "ext",
34098            attribute : "qtip",
34099            width : "width",
34100            target : "target",
34101            title : "qtitle",
34102            hide : "hide",
34103            cls : "qclass"
34104        }
34105    };
34106 }();
34107
34108 // backwards compat
34109 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34110  * Based on:
34111  * Ext JS Library 1.1.1
34112  * Copyright(c) 2006-2007, Ext JS, LLC.
34113  *
34114  * Originally Released Under LGPL - original licence link has changed is not relivant.
34115  *
34116  * Fork - LGPL
34117  * <script type="text/javascript">
34118  */
34119  
34120
34121 /**
34122  * @class Roo.tree.TreePanel
34123  * @extends Roo.data.Tree
34124
34125  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34126  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34127  * @cfg {Boolean} enableDD true to enable drag and drop
34128  * @cfg {Boolean} enableDrag true to enable just drag
34129  * @cfg {Boolean} enableDrop true to enable just drop
34130  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34131  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34132  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34133  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34134  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34135  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34136  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34137  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34138  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34139  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34140  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34141  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34142  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34143  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34144  * @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>
34145  * @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>
34146  * 
34147  * @constructor
34148  * @param {String/HTMLElement/Element} el The container element
34149  * @param {Object} config
34150  */
34151 Roo.tree.TreePanel = function(el, config){
34152     var root = false;
34153     var loader = false;
34154     if (config.root) {
34155         root = config.root;
34156         delete config.root;
34157     }
34158     if (config.loader) {
34159         loader = config.loader;
34160         delete config.loader;
34161     }
34162     
34163     Roo.apply(this, config);
34164     Roo.tree.TreePanel.superclass.constructor.call(this);
34165     this.el = Roo.get(el);
34166     this.el.addClass('x-tree');
34167     //console.log(root);
34168     if (root) {
34169         this.setRootNode( Roo.factory(root, Roo.tree));
34170     }
34171     if (loader) {
34172         this.loader = Roo.factory(loader, Roo.tree);
34173     }
34174    /**
34175     * Read-only. The id of the container element becomes this TreePanel's id.
34176     */
34177     this.id = this.el.id;
34178     this.addEvents({
34179         /**
34180         * @event beforeload
34181         * Fires before a node is loaded, return false to cancel
34182         * @param {Node} node The node being loaded
34183         */
34184         "beforeload" : true,
34185         /**
34186         * @event load
34187         * Fires when a node is loaded
34188         * @param {Node} node The node that was loaded
34189         */
34190         "load" : true,
34191         /**
34192         * @event textchange
34193         * Fires when the text for a node is changed
34194         * @param {Node} node The node
34195         * @param {String} text The new text
34196         * @param {String} oldText The old text
34197         */
34198         "textchange" : true,
34199         /**
34200         * @event beforeexpand
34201         * Fires before a node is expanded, return false to cancel.
34202         * @param {Node} node The node
34203         * @param {Boolean} deep
34204         * @param {Boolean} anim
34205         */
34206         "beforeexpand" : true,
34207         /**
34208         * @event beforecollapse
34209         * Fires before a node is collapsed, return false to cancel.
34210         * @param {Node} node The node
34211         * @param {Boolean} deep
34212         * @param {Boolean} anim
34213         */
34214         "beforecollapse" : true,
34215         /**
34216         * @event expand
34217         * Fires when a node is expanded
34218         * @param {Node} node The node
34219         */
34220         "expand" : true,
34221         /**
34222         * @event disabledchange
34223         * Fires when the disabled status of a node changes
34224         * @param {Node} node The node
34225         * @param {Boolean} disabled
34226         */
34227         "disabledchange" : true,
34228         /**
34229         * @event collapse
34230         * Fires when a node is collapsed
34231         * @param {Node} node The node
34232         */
34233         "collapse" : true,
34234         /**
34235         * @event beforeclick
34236         * Fires before click processing on a node. Return false to cancel the default action.
34237         * @param {Node} node The node
34238         * @param {Roo.EventObject} e The event object
34239         */
34240         "beforeclick":true,
34241         /**
34242         * @event checkchange
34243         * Fires when a node with a checkbox's checked property changes
34244         * @param {Node} this This node
34245         * @param {Boolean} checked
34246         */
34247         "checkchange":true,
34248         /**
34249         * @event click
34250         * Fires when a node is clicked
34251         * @param {Node} node The node
34252         * @param {Roo.EventObject} e The event object
34253         */
34254         "click":true,
34255         /**
34256         * @event dblclick
34257         * Fires when a node is double clicked
34258         * @param {Node} node The node
34259         * @param {Roo.EventObject} e The event object
34260         */
34261         "dblclick":true,
34262         /**
34263         * @event contextmenu
34264         * Fires when a node is right clicked
34265         * @param {Node} node The node
34266         * @param {Roo.EventObject} e The event object
34267         */
34268         "contextmenu":true,
34269         /**
34270         * @event beforechildrenrendered
34271         * Fires right before the child nodes for a node are rendered
34272         * @param {Node} node The node
34273         */
34274         "beforechildrenrendered":true,
34275         /**
34276         * @event startdrag
34277         * Fires when a node starts being dragged
34278         * @param {Roo.tree.TreePanel} this
34279         * @param {Roo.tree.TreeNode} node
34280         * @param {event} e The raw browser event
34281         */ 
34282        "startdrag" : true,
34283        /**
34284         * @event enddrag
34285         * Fires when a drag operation is complete
34286         * @param {Roo.tree.TreePanel} this
34287         * @param {Roo.tree.TreeNode} node
34288         * @param {event} e The raw browser event
34289         */
34290        "enddrag" : true,
34291        /**
34292         * @event dragdrop
34293         * Fires when a dragged node is dropped on a valid DD target
34294         * @param {Roo.tree.TreePanel} this
34295         * @param {Roo.tree.TreeNode} node
34296         * @param {DD} dd The dd it was dropped on
34297         * @param {event} e The raw browser event
34298         */
34299        "dragdrop" : true,
34300        /**
34301         * @event beforenodedrop
34302         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34303         * passed to handlers has the following properties:<br />
34304         * <ul style="padding:5px;padding-left:16px;">
34305         * <li>tree - The TreePanel</li>
34306         * <li>target - The node being targeted for the drop</li>
34307         * <li>data - The drag data from the drag source</li>
34308         * <li>point - The point of the drop - append, above or below</li>
34309         * <li>source - The drag source</li>
34310         * <li>rawEvent - Raw mouse event</li>
34311         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34312         * to be inserted by setting them on this object.</li>
34313         * <li>cancel - Set this to true to cancel the drop.</li>
34314         * </ul>
34315         * @param {Object} dropEvent
34316         */
34317        "beforenodedrop" : true,
34318        /**
34319         * @event nodedrop
34320         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34321         * passed to handlers has the following properties:<br />
34322         * <ul style="padding:5px;padding-left:16px;">
34323         * <li>tree - The TreePanel</li>
34324         * <li>target - The node being targeted for the drop</li>
34325         * <li>data - The drag data from the drag source</li>
34326         * <li>point - The point of the drop - append, above or below</li>
34327         * <li>source - The drag source</li>
34328         * <li>rawEvent - Raw mouse event</li>
34329         * <li>dropNode - Dropped node(s).</li>
34330         * </ul>
34331         * @param {Object} dropEvent
34332         */
34333        "nodedrop" : true,
34334         /**
34335         * @event nodedragover
34336         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34337         * passed to handlers has the following properties:<br />
34338         * <ul style="padding:5px;padding-left:16px;">
34339         * <li>tree - The TreePanel</li>
34340         * <li>target - The node being targeted for the drop</li>
34341         * <li>data - The drag data from the drag source</li>
34342         * <li>point - The point of the drop - append, above or below</li>
34343         * <li>source - The drag source</li>
34344         * <li>rawEvent - Raw mouse event</li>
34345         * <li>dropNode - Drop node(s) provided by the source.</li>
34346         * <li>cancel - Set this to true to signal drop not allowed.</li>
34347         * </ul>
34348         * @param {Object} dragOverEvent
34349         */
34350        "nodedragover" : true,
34351        /**
34352         * @event appendnode
34353         * Fires when append node to the tree
34354         * @param {Roo.tree.TreePanel} this
34355         * @param {Roo.tree.TreeNode} node
34356         * @param {Number} index The index of the newly appended node
34357         */
34358        "appendnode" : true
34359         
34360     });
34361     if(this.singleExpand){
34362        this.on("beforeexpand", this.restrictExpand, this);
34363     }
34364     if (this.editor) {
34365         this.editor.tree = this;
34366         this.editor = Roo.factory(this.editor, Roo.tree);
34367     }
34368     
34369     if (this.selModel) {
34370         this.selModel = Roo.factory(this.selModel, Roo.tree);
34371     }
34372    
34373 };
34374 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34375     rootVisible : true,
34376     animate: Roo.enableFx,
34377     lines : true,
34378     enableDD : false,
34379     hlDrop : Roo.enableFx,
34380   
34381     renderer: false,
34382     
34383     rendererTip: false,
34384     // private
34385     restrictExpand : function(node){
34386         var p = node.parentNode;
34387         if(p){
34388             if(p.expandedChild && p.expandedChild.parentNode == p){
34389                 p.expandedChild.collapse();
34390             }
34391             p.expandedChild = node;
34392         }
34393     },
34394
34395     // private override
34396     setRootNode : function(node){
34397         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34398         if(!this.rootVisible){
34399             node.ui = new Roo.tree.RootTreeNodeUI(node);
34400         }
34401         return node;
34402     },
34403
34404     /**
34405      * Returns the container element for this TreePanel
34406      */
34407     getEl : function(){
34408         return this.el;
34409     },
34410
34411     /**
34412      * Returns the default TreeLoader for this TreePanel
34413      */
34414     getLoader : function(){
34415         return this.loader;
34416     },
34417
34418     /**
34419      * Expand all nodes
34420      */
34421     expandAll : function(){
34422         this.root.expand(true);
34423     },
34424
34425     /**
34426      * Collapse all nodes
34427      */
34428     collapseAll : function(){
34429         this.root.collapse(true);
34430     },
34431
34432     /**
34433      * Returns the selection model used by this TreePanel
34434      */
34435     getSelectionModel : function(){
34436         if(!this.selModel){
34437             this.selModel = new Roo.tree.DefaultSelectionModel();
34438         }
34439         return this.selModel;
34440     },
34441
34442     /**
34443      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34444      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34445      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34446      * @return {Array}
34447      */
34448     getChecked : function(a, startNode){
34449         startNode = startNode || this.root;
34450         var r = [];
34451         var f = function(){
34452             if(this.attributes.checked){
34453                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34454             }
34455         }
34456         startNode.cascade(f);
34457         return r;
34458     },
34459
34460     /**
34461      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34462      * @param {String} path
34463      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34464      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34465      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34466      */
34467     expandPath : function(path, attr, callback){
34468         attr = attr || "id";
34469         var keys = path.split(this.pathSeparator);
34470         var curNode = this.root;
34471         if(curNode.attributes[attr] != keys[1]){ // invalid root
34472             if(callback){
34473                 callback(false, null);
34474             }
34475             return;
34476         }
34477         var index = 1;
34478         var f = function(){
34479             if(++index == keys.length){
34480                 if(callback){
34481                     callback(true, curNode);
34482                 }
34483                 return;
34484             }
34485             var c = curNode.findChild(attr, keys[index]);
34486             if(!c){
34487                 if(callback){
34488                     callback(false, curNode);
34489                 }
34490                 return;
34491             }
34492             curNode = c;
34493             c.expand(false, false, f);
34494         };
34495         curNode.expand(false, false, f);
34496     },
34497
34498     /**
34499      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34500      * @param {String} path
34501      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34502      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34503      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34504      */
34505     selectPath : function(path, attr, callback){
34506         attr = attr || "id";
34507         var keys = path.split(this.pathSeparator);
34508         var v = keys.pop();
34509         if(keys.length > 0){
34510             var f = function(success, node){
34511                 if(success && node){
34512                     var n = node.findChild(attr, v);
34513                     if(n){
34514                         n.select();
34515                         if(callback){
34516                             callback(true, n);
34517                         }
34518                     }else if(callback){
34519                         callback(false, n);
34520                     }
34521                 }else{
34522                     if(callback){
34523                         callback(false, n);
34524                     }
34525                 }
34526             };
34527             this.expandPath(keys.join(this.pathSeparator), attr, f);
34528         }else{
34529             this.root.select();
34530             if(callback){
34531                 callback(true, this.root);
34532             }
34533         }
34534     },
34535
34536     getTreeEl : function(){
34537         return this.el;
34538     },
34539
34540     /**
34541      * Trigger rendering of this TreePanel
34542      */
34543     render : function(){
34544         if (this.innerCt) {
34545             return this; // stop it rendering more than once!!
34546         }
34547         
34548         this.innerCt = this.el.createChild({tag:"ul",
34549                cls:"x-tree-root-ct " +
34550                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34551
34552         if(this.containerScroll){
34553             Roo.dd.ScrollManager.register(this.el);
34554         }
34555         if((this.enableDD || this.enableDrop) && !this.dropZone){
34556            /**
34557             * The dropZone used by this tree if drop is enabled
34558             * @type Roo.tree.TreeDropZone
34559             */
34560              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34561                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34562            });
34563         }
34564         if((this.enableDD || this.enableDrag) && !this.dragZone){
34565            /**
34566             * The dragZone used by this tree if drag is enabled
34567             * @type Roo.tree.TreeDragZone
34568             */
34569             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34570                ddGroup: this.ddGroup || "TreeDD",
34571                scroll: this.ddScroll
34572            });
34573         }
34574         this.getSelectionModel().init(this);
34575         if (!this.root) {
34576             Roo.log("ROOT not set in tree");
34577             return this;
34578         }
34579         this.root.render();
34580         if(!this.rootVisible){
34581             this.root.renderChildren();
34582         }
34583         return this;
34584     }
34585 });/*
34586  * Based on:
34587  * Ext JS Library 1.1.1
34588  * Copyright(c) 2006-2007, Ext JS, LLC.
34589  *
34590  * Originally Released Under LGPL - original licence link has changed is not relivant.
34591  *
34592  * Fork - LGPL
34593  * <script type="text/javascript">
34594  */
34595  
34596
34597 /**
34598  * @class Roo.tree.DefaultSelectionModel
34599  * @extends Roo.util.Observable
34600  * The default single selection for a TreePanel.
34601  * @param {Object} cfg Configuration
34602  */
34603 Roo.tree.DefaultSelectionModel = function(cfg){
34604    this.selNode = null;
34605    
34606    
34607    
34608    this.addEvents({
34609        /**
34610         * @event selectionchange
34611         * Fires when the selected node changes
34612         * @param {DefaultSelectionModel} this
34613         * @param {TreeNode} node the new selection
34614         */
34615        "selectionchange" : true,
34616
34617        /**
34618         * @event beforeselect
34619         * Fires before the selected node changes, return false to cancel the change
34620         * @param {DefaultSelectionModel} this
34621         * @param {TreeNode} node the new selection
34622         * @param {TreeNode} node the old selection
34623         */
34624        "beforeselect" : true
34625    });
34626    
34627     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34628 };
34629
34630 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34631     init : function(tree){
34632         this.tree = tree;
34633         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34634         tree.on("click", this.onNodeClick, this);
34635     },
34636     
34637     onNodeClick : function(node, e){
34638         if (e.ctrlKey && this.selNode == node)  {
34639             this.unselect(node);
34640             return;
34641         }
34642         this.select(node);
34643     },
34644     
34645     /**
34646      * Select a node.
34647      * @param {TreeNode} node The node to select
34648      * @return {TreeNode} The selected node
34649      */
34650     select : function(node){
34651         var last = this.selNode;
34652         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34653             if(last){
34654                 last.ui.onSelectedChange(false);
34655             }
34656             this.selNode = node;
34657             node.ui.onSelectedChange(true);
34658             this.fireEvent("selectionchange", this, node, last);
34659         }
34660         return node;
34661     },
34662     
34663     /**
34664      * Deselect a node.
34665      * @param {TreeNode} node The node to unselect
34666      */
34667     unselect : function(node){
34668         if(this.selNode == node){
34669             this.clearSelections();
34670         }    
34671     },
34672     
34673     /**
34674      * Clear all selections
34675      */
34676     clearSelections : function(){
34677         var n = this.selNode;
34678         if(n){
34679             n.ui.onSelectedChange(false);
34680             this.selNode = null;
34681             this.fireEvent("selectionchange", this, null);
34682         }
34683         return n;
34684     },
34685     
34686     /**
34687      * Get the selected node
34688      * @return {TreeNode} The selected node
34689      */
34690     getSelectedNode : function(){
34691         return this.selNode;    
34692     },
34693     
34694     /**
34695      * Returns true if the node is selected
34696      * @param {TreeNode} node The node to check
34697      * @return {Boolean}
34698      */
34699     isSelected : function(node){
34700         return this.selNode == node;  
34701     },
34702
34703     /**
34704      * Selects the node above the selected node in the tree, intelligently walking the nodes
34705      * @return TreeNode The new selection
34706      */
34707     selectPrevious : function(){
34708         var s = this.selNode || this.lastSelNode;
34709         if(!s){
34710             return null;
34711         }
34712         var ps = s.previousSibling;
34713         if(ps){
34714             if(!ps.isExpanded() || ps.childNodes.length < 1){
34715                 return this.select(ps);
34716             } else{
34717                 var lc = ps.lastChild;
34718                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34719                     lc = lc.lastChild;
34720                 }
34721                 return this.select(lc);
34722             }
34723         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34724             return this.select(s.parentNode);
34725         }
34726         return null;
34727     },
34728
34729     /**
34730      * Selects the node above the selected node in the tree, intelligently walking the nodes
34731      * @return TreeNode The new selection
34732      */
34733     selectNext : function(){
34734         var s = this.selNode || this.lastSelNode;
34735         if(!s){
34736             return null;
34737         }
34738         if(s.firstChild && s.isExpanded()){
34739              return this.select(s.firstChild);
34740          }else if(s.nextSibling){
34741              return this.select(s.nextSibling);
34742          }else if(s.parentNode){
34743             var newS = null;
34744             s.parentNode.bubble(function(){
34745                 if(this.nextSibling){
34746                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34747                     return false;
34748                 }
34749             });
34750             return newS;
34751          }
34752         return null;
34753     },
34754
34755     onKeyDown : function(e){
34756         var s = this.selNode || this.lastSelNode;
34757         // undesirable, but required
34758         var sm = this;
34759         if(!s){
34760             return;
34761         }
34762         var k = e.getKey();
34763         switch(k){
34764              case e.DOWN:
34765                  e.stopEvent();
34766                  this.selectNext();
34767              break;
34768              case e.UP:
34769                  e.stopEvent();
34770                  this.selectPrevious();
34771              break;
34772              case e.RIGHT:
34773                  e.preventDefault();
34774                  if(s.hasChildNodes()){
34775                      if(!s.isExpanded()){
34776                          s.expand();
34777                      }else if(s.firstChild){
34778                          this.select(s.firstChild, e);
34779                      }
34780                  }
34781              break;
34782              case e.LEFT:
34783                  e.preventDefault();
34784                  if(s.hasChildNodes() && s.isExpanded()){
34785                      s.collapse();
34786                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34787                      this.select(s.parentNode, e);
34788                  }
34789              break;
34790         };
34791     }
34792 });
34793
34794 /**
34795  * @class Roo.tree.MultiSelectionModel
34796  * @extends Roo.util.Observable
34797  * Multi selection for a TreePanel.
34798  * @param {Object} cfg Configuration
34799  */
34800 Roo.tree.MultiSelectionModel = function(){
34801    this.selNodes = [];
34802    this.selMap = {};
34803    this.addEvents({
34804        /**
34805         * @event selectionchange
34806         * Fires when the selected nodes change
34807         * @param {MultiSelectionModel} this
34808         * @param {Array} nodes Array of the selected nodes
34809         */
34810        "selectionchange" : true
34811    });
34812    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34813    
34814 };
34815
34816 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34817     init : function(tree){
34818         this.tree = tree;
34819         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34820         tree.on("click", this.onNodeClick, this);
34821     },
34822     
34823     onNodeClick : function(node, e){
34824         this.select(node, e, e.ctrlKey);
34825     },
34826     
34827     /**
34828      * Select a node.
34829      * @param {TreeNode} node The node to select
34830      * @param {EventObject} e (optional) An event associated with the selection
34831      * @param {Boolean} keepExisting True to retain existing selections
34832      * @return {TreeNode} The selected node
34833      */
34834     select : function(node, e, keepExisting){
34835         if(keepExisting !== true){
34836             this.clearSelections(true);
34837         }
34838         if(this.isSelected(node)){
34839             this.lastSelNode = node;
34840             return node;
34841         }
34842         this.selNodes.push(node);
34843         this.selMap[node.id] = node;
34844         this.lastSelNode = node;
34845         node.ui.onSelectedChange(true);
34846         this.fireEvent("selectionchange", this, this.selNodes);
34847         return node;
34848     },
34849     
34850     /**
34851      * Deselect a node.
34852      * @param {TreeNode} node The node to unselect
34853      */
34854     unselect : function(node){
34855         if(this.selMap[node.id]){
34856             node.ui.onSelectedChange(false);
34857             var sn = this.selNodes;
34858             var index = -1;
34859             if(sn.indexOf){
34860                 index = sn.indexOf(node);
34861             }else{
34862                 for(var i = 0, len = sn.length; i < len; i++){
34863                     if(sn[i] == node){
34864                         index = i;
34865                         break;
34866                     }
34867                 }
34868             }
34869             if(index != -1){
34870                 this.selNodes.splice(index, 1);
34871             }
34872             delete this.selMap[node.id];
34873             this.fireEvent("selectionchange", this, this.selNodes);
34874         }
34875     },
34876     
34877     /**
34878      * Clear all selections
34879      */
34880     clearSelections : function(suppressEvent){
34881         var sn = this.selNodes;
34882         if(sn.length > 0){
34883             for(var i = 0, len = sn.length; i < len; i++){
34884                 sn[i].ui.onSelectedChange(false);
34885             }
34886             this.selNodes = [];
34887             this.selMap = {};
34888             if(suppressEvent !== true){
34889                 this.fireEvent("selectionchange", this, this.selNodes);
34890             }
34891         }
34892     },
34893     
34894     /**
34895      * Returns true if the node is selected
34896      * @param {TreeNode} node The node to check
34897      * @return {Boolean}
34898      */
34899     isSelected : function(node){
34900         return this.selMap[node.id] ? true : false;  
34901     },
34902     
34903     /**
34904      * Returns an array of the selected nodes
34905      * @return {Array}
34906      */
34907     getSelectedNodes : function(){
34908         return this.selNodes;    
34909     },
34910
34911     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34912
34913     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34914
34915     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34916 });/*
34917  * Based on:
34918  * Ext JS Library 1.1.1
34919  * Copyright(c) 2006-2007, Ext JS, LLC.
34920  *
34921  * Originally Released Under LGPL - original licence link has changed is not relivant.
34922  *
34923  * Fork - LGPL
34924  * <script type="text/javascript">
34925  */
34926  
34927 /**
34928  * @class Roo.tree.TreeNode
34929  * @extends Roo.data.Node
34930  * @cfg {String} text The text for this node
34931  * @cfg {Boolean} expanded true to start the node expanded
34932  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34933  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34934  * @cfg {Boolean} disabled true to start the node disabled
34935  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34936  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34937  * @cfg {String} cls A css class to be added to the node
34938  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34939  * @cfg {String} href URL of the link used for the node (defaults to #)
34940  * @cfg {String} hrefTarget target frame for the link
34941  * @cfg {String} qtip An Ext QuickTip for the node
34942  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34943  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34944  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34945  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34946  * (defaults to undefined with no checkbox rendered)
34947  * @constructor
34948  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34949  */
34950 Roo.tree.TreeNode = function(attributes){
34951     attributes = attributes || {};
34952     if(typeof attributes == "string"){
34953         attributes = {text: attributes};
34954     }
34955     this.childrenRendered = false;
34956     this.rendered = false;
34957     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34958     this.expanded = attributes.expanded === true;
34959     this.isTarget = attributes.isTarget !== false;
34960     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34961     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34962
34963     /**
34964      * Read-only. The text for this node. To change it use setText().
34965      * @type String
34966      */
34967     this.text = attributes.text;
34968     /**
34969      * True if this node is disabled.
34970      * @type Boolean
34971      */
34972     this.disabled = attributes.disabled === true;
34973
34974     this.addEvents({
34975         /**
34976         * @event textchange
34977         * Fires when the text for this node is changed
34978         * @param {Node} this This node
34979         * @param {String} text The new text
34980         * @param {String} oldText The old text
34981         */
34982         "textchange" : true,
34983         /**
34984         * @event beforeexpand
34985         * Fires before this node is expanded, return false to cancel.
34986         * @param {Node} this This node
34987         * @param {Boolean} deep
34988         * @param {Boolean} anim
34989         */
34990         "beforeexpand" : true,
34991         /**
34992         * @event beforecollapse
34993         * Fires before this node is collapsed, return false to cancel.
34994         * @param {Node} this This node
34995         * @param {Boolean} deep
34996         * @param {Boolean} anim
34997         */
34998         "beforecollapse" : true,
34999         /**
35000         * @event expand
35001         * Fires when this node is expanded
35002         * @param {Node} this This node
35003         */
35004         "expand" : true,
35005         /**
35006         * @event disabledchange
35007         * Fires when the disabled status of this node changes
35008         * @param {Node} this This node
35009         * @param {Boolean} disabled
35010         */
35011         "disabledchange" : true,
35012         /**
35013         * @event collapse
35014         * Fires when this node is collapsed
35015         * @param {Node} this This node
35016         */
35017         "collapse" : true,
35018         /**
35019         * @event beforeclick
35020         * Fires before click processing. Return false to cancel the default action.
35021         * @param {Node} this This node
35022         * @param {Roo.EventObject} e The event object
35023         */
35024         "beforeclick":true,
35025         /**
35026         * @event checkchange
35027         * Fires when a node with a checkbox's checked property changes
35028         * @param {Node} this This node
35029         * @param {Boolean} checked
35030         */
35031         "checkchange":true,
35032         /**
35033         * @event click
35034         * Fires when this node is clicked
35035         * @param {Node} this This node
35036         * @param {Roo.EventObject} e The event object
35037         */
35038         "click":true,
35039         /**
35040         * @event dblclick
35041         * Fires when this node is double clicked
35042         * @param {Node} this This node
35043         * @param {Roo.EventObject} e The event object
35044         */
35045         "dblclick":true,
35046         /**
35047         * @event contextmenu
35048         * Fires when this node is right clicked
35049         * @param {Node} this This node
35050         * @param {Roo.EventObject} e The event object
35051         */
35052         "contextmenu":true,
35053         /**
35054         * @event beforechildrenrendered
35055         * Fires right before the child nodes for this node are rendered
35056         * @param {Node} this This node
35057         */
35058         "beforechildrenrendered":true
35059     });
35060
35061     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35062
35063     /**
35064      * Read-only. The UI for this node
35065      * @type TreeNodeUI
35066      */
35067     this.ui = new uiClass(this);
35068     
35069     // finally support items[]
35070     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35071         return;
35072     }
35073     
35074     
35075     Roo.each(this.attributes.items, function(c) {
35076         this.appendChild(Roo.factory(c,Roo.Tree));
35077     }, this);
35078     delete this.attributes.items;
35079     
35080     
35081     
35082 };
35083 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35084     preventHScroll: true,
35085     /**
35086      * Returns true if this node is expanded
35087      * @return {Boolean}
35088      */
35089     isExpanded : function(){
35090         return this.expanded;
35091     },
35092
35093     /**
35094      * Returns the UI object for this node
35095      * @return {TreeNodeUI}
35096      */
35097     getUI : function(){
35098         return this.ui;
35099     },
35100
35101     // private override
35102     setFirstChild : function(node){
35103         var of = this.firstChild;
35104         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35105         if(this.childrenRendered && of && node != of){
35106             of.renderIndent(true, true);
35107         }
35108         if(this.rendered){
35109             this.renderIndent(true, true);
35110         }
35111     },
35112
35113     // private override
35114     setLastChild : function(node){
35115         var ol = this.lastChild;
35116         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35117         if(this.childrenRendered && ol && node != ol){
35118             ol.renderIndent(true, true);
35119         }
35120         if(this.rendered){
35121             this.renderIndent(true, true);
35122         }
35123     },
35124
35125     // these methods are overridden to provide lazy rendering support
35126     // private override
35127     appendChild : function()
35128     {
35129         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35130         if(node && this.childrenRendered){
35131             node.render();
35132         }
35133         this.ui.updateExpandIcon();
35134         return node;
35135     },
35136
35137     // private override
35138     removeChild : function(node){
35139         this.ownerTree.getSelectionModel().unselect(node);
35140         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35141         // if it's been rendered remove dom node
35142         if(this.childrenRendered){
35143             node.ui.remove();
35144         }
35145         if(this.childNodes.length < 1){
35146             this.collapse(false, false);
35147         }else{
35148             this.ui.updateExpandIcon();
35149         }
35150         if(!this.firstChild) {
35151             this.childrenRendered = false;
35152         }
35153         return node;
35154     },
35155
35156     // private override
35157     insertBefore : function(node, refNode){
35158         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35159         if(newNode && refNode && this.childrenRendered){
35160             node.render();
35161         }
35162         this.ui.updateExpandIcon();
35163         return newNode;
35164     },
35165
35166     /**
35167      * Sets the text for this node
35168      * @param {String} text
35169      */
35170     setText : function(text){
35171         var oldText = this.text;
35172         this.text = text;
35173         this.attributes.text = text;
35174         if(this.rendered){ // event without subscribing
35175             this.ui.onTextChange(this, text, oldText);
35176         }
35177         this.fireEvent("textchange", this, text, oldText);
35178     },
35179
35180     /**
35181      * Triggers selection of this node
35182      */
35183     select : function(){
35184         this.getOwnerTree().getSelectionModel().select(this);
35185     },
35186
35187     /**
35188      * Triggers deselection of this node
35189      */
35190     unselect : function(){
35191         this.getOwnerTree().getSelectionModel().unselect(this);
35192     },
35193
35194     /**
35195      * Returns true if this node is selected
35196      * @return {Boolean}
35197      */
35198     isSelected : function(){
35199         return this.getOwnerTree().getSelectionModel().isSelected(this);
35200     },
35201
35202     /**
35203      * Expand this node.
35204      * @param {Boolean} deep (optional) True to expand all children as well
35205      * @param {Boolean} anim (optional) false to cancel the default animation
35206      * @param {Function} callback (optional) A callback to be called when
35207      * expanding this node completes (does not wait for deep expand to complete).
35208      * Called with 1 parameter, this node.
35209      */
35210     expand : function(deep, anim, callback){
35211         if(!this.expanded){
35212             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35213                 return;
35214             }
35215             if(!this.childrenRendered){
35216                 this.renderChildren();
35217             }
35218             this.expanded = true;
35219             
35220             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35221                 this.ui.animExpand(function(){
35222                     this.fireEvent("expand", this);
35223                     if(typeof callback == "function"){
35224                         callback(this);
35225                     }
35226                     if(deep === true){
35227                         this.expandChildNodes(true);
35228                     }
35229                 }.createDelegate(this));
35230                 return;
35231             }else{
35232                 this.ui.expand();
35233                 this.fireEvent("expand", this);
35234                 if(typeof callback == "function"){
35235                     callback(this);
35236                 }
35237             }
35238         }else{
35239            if(typeof callback == "function"){
35240                callback(this);
35241            }
35242         }
35243         if(deep === true){
35244             this.expandChildNodes(true);
35245         }
35246     },
35247
35248     isHiddenRoot : function(){
35249         return this.isRoot && !this.getOwnerTree().rootVisible;
35250     },
35251
35252     /**
35253      * Collapse this node.
35254      * @param {Boolean} deep (optional) True to collapse all children as well
35255      * @param {Boolean} anim (optional) false to cancel the default animation
35256      */
35257     collapse : function(deep, anim){
35258         if(this.expanded && !this.isHiddenRoot()){
35259             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35260                 return;
35261             }
35262             this.expanded = false;
35263             if((this.getOwnerTree().animate && anim !== false) || anim){
35264                 this.ui.animCollapse(function(){
35265                     this.fireEvent("collapse", this);
35266                     if(deep === true){
35267                         this.collapseChildNodes(true);
35268                     }
35269                 }.createDelegate(this));
35270                 return;
35271             }else{
35272                 this.ui.collapse();
35273                 this.fireEvent("collapse", this);
35274             }
35275         }
35276         if(deep === true){
35277             var cs = this.childNodes;
35278             for(var i = 0, len = cs.length; i < len; i++) {
35279                 cs[i].collapse(true, false);
35280             }
35281         }
35282     },
35283
35284     // private
35285     delayedExpand : function(delay){
35286         if(!this.expandProcId){
35287             this.expandProcId = this.expand.defer(delay, this);
35288         }
35289     },
35290
35291     // private
35292     cancelExpand : function(){
35293         if(this.expandProcId){
35294             clearTimeout(this.expandProcId);
35295         }
35296         this.expandProcId = false;
35297     },
35298
35299     /**
35300      * Toggles expanded/collapsed state of the node
35301      */
35302     toggle : function(){
35303         if(this.expanded){
35304             this.collapse();
35305         }else{
35306             this.expand();
35307         }
35308     },
35309
35310     /**
35311      * Ensures all parent nodes are expanded
35312      */
35313     ensureVisible : function(callback){
35314         var tree = this.getOwnerTree();
35315         tree.expandPath(this.parentNode.getPath(), false, function(){
35316             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35317             Roo.callback(callback);
35318         }.createDelegate(this));
35319     },
35320
35321     /**
35322      * Expand all child nodes
35323      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35324      */
35325     expandChildNodes : function(deep){
35326         var cs = this.childNodes;
35327         for(var i = 0, len = cs.length; i < len; i++) {
35328                 cs[i].expand(deep);
35329         }
35330     },
35331
35332     /**
35333      * Collapse all child nodes
35334      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35335      */
35336     collapseChildNodes : function(deep){
35337         var cs = this.childNodes;
35338         for(var i = 0, len = cs.length; i < len; i++) {
35339                 cs[i].collapse(deep);
35340         }
35341     },
35342
35343     /**
35344      * Disables this node
35345      */
35346     disable : function(){
35347         this.disabled = true;
35348         this.unselect();
35349         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35350             this.ui.onDisableChange(this, true);
35351         }
35352         this.fireEvent("disabledchange", this, true);
35353     },
35354
35355     /**
35356      * Enables this node
35357      */
35358     enable : function(){
35359         this.disabled = false;
35360         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35361             this.ui.onDisableChange(this, false);
35362         }
35363         this.fireEvent("disabledchange", this, false);
35364     },
35365
35366     // private
35367     renderChildren : function(suppressEvent){
35368         if(suppressEvent !== false){
35369             this.fireEvent("beforechildrenrendered", this);
35370         }
35371         var cs = this.childNodes;
35372         for(var i = 0, len = cs.length; i < len; i++){
35373             cs[i].render(true);
35374         }
35375         this.childrenRendered = true;
35376     },
35377
35378     // private
35379     sort : function(fn, scope){
35380         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35381         if(this.childrenRendered){
35382             var cs = this.childNodes;
35383             for(var i = 0, len = cs.length; i < len; i++){
35384                 cs[i].render(true);
35385             }
35386         }
35387     },
35388
35389     // private
35390     render : function(bulkRender){
35391         this.ui.render(bulkRender);
35392         if(!this.rendered){
35393             this.rendered = true;
35394             if(this.expanded){
35395                 this.expanded = false;
35396                 this.expand(false, false);
35397             }
35398         }
35399     },
35400
35401     // private
35402     renderIndent : function(deep, refresh){
35403         if(refresh){
35404             this.ui.childIndent = null;
35405         }
35406         this.ui.renderIndent();
35407         if(deep === true && this.childrenRendered){
35408             var cs = this.childNodes;
35409             for(var i = 0, len = cs.length; i < len; i++){
35410                 cs[i].renderIndent(true, refresh);
35411             }
35412         }
35413     }
35414 });/*
35415  * Based on:
35416  * Ext JS Library 1.1.1
35417  * Copyright(c) 2006-2007, Ext JS, LLC.
35418  *
35419  * Originally Released Under LGPL - original licence link has changed is not relivant.
35420  *
35421  * Fork - LGPL
35422  * <script type="text/javascript">
35423  */
35424  
35425 /**
35426  * @class Roo.tree.AsyncTreeNode
35427  * @extends Roo.tree.TreeNode
35428  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35429  * @constructor
35430  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35431  */
35432  Roo.tree.AsyncTreeNode = function(config){
35433     this.loaded = false;
35434     this.loading = false;
35435     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35436     /**
35437     * @event beforeload
35438     * Fires before this node is loaded, return false to cancel
35439     * @param {Node} this This node
35440     */
35441     this.addEvents({'beforeload':true, 'load': true});
35442     /**
35443     * @event load
35444     * Fires when this node is loaded
35445     * @param {Node} this This node
35446     */
35447     /**
35448      * The loader used by this node (defaults to using the tree's defined loader)
35449      * @type TreeLoader
35450      * @property loader
35451      */
35452 };
35453 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35454     expand : function(deep, anim, callback){
35455         if(this.loading){ // if an async load is already running, waiting til it's done
35456             var timer;
35457             var f = function(){
35458                 if(!this.loading){ // done loading
35459                     clearInterval(timer);
35460                     this.expand(deep, anim, callback);
35461                 }
35462             }.createDelegate(this);
35463             timer = setInterval(f, 200);
35464             return;
35465         }
35466         if(!this.loaded){
35467             if(this.fireEvent("beforeload", this) === false){
35468                 return;
35469             }
35470             this.loading = true;
35471             this.ui.beforeLoad(this);
35472             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35473             if(loader){
35474                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35475                 return;
35476             }
35477         }
35478         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35479     },
35480     
35481     /**
35482      * Returns true if this node is currently loading
35483      * @return {Boolean}
35484      */
35485     isLoading : function(){
35486         return this.loading;  
35487     },
35488     
35489     loadComplete : function(deep, anim, callback){
35490         this.loading = false;
35491         this.loaded = true;
35492         this.ui.afterLoad(this);
35493         this.fireEvent("load", this);
35494         this.expand(deep, anim, callback);
35495     },
35496     
35497     /**
35498      * Returns true if this node has been loaded
35499      * @return {Boolean}
35500      */
35501     isLoaded : function(){
35502         return this.loaded;
35503     },
35504     
35505     hasChildNodes : function(){
35506         if(!this.isLeaf() && !this.loaded){
35507             return true;
35508         }else{
35509             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35510         }
35511     },
35512
35513     /**
35514      * Trigger a reload for this node
35515      * @param {Function} callback
35516      */
35517     reload : function(callback){
35518         this.collapse(false, false);
35519         while(this.firstChild){
35520             this.removeChild(this.firstChild);
35521         }
35522         this.childrenRendered = false;
35523         this.loaded = false;
35524         if(this.isHiddenRoot()){
35525             this.expanded = false;
35526         }
35527         this.expand(false, false, callback);
35528     }
35529 });/*
35530  * Based on:
35531  * Ext JS Library 1.1.1
35532  * Copyright(c) 2006-2007, Ext JS, LLC.
35533  *
35534  * Originally Released Under LGPL - original licence link has changed is not relivant.
35535  *
35536  * Fork - LGPL
35537  * <script type="text/javascript">
35538  */
35539  
35540 /**
35541  * @class Roo.tree.TreeNodeUI
35542  * @constructor
35543  * @param {Object} node The node to render
35544  * The TreeNode UI implementation is separate from the
35545  * tree implementation. Unless you are customizing the tree UI,
35546  * you should never have to use this directly.
35547  */
35548 Roo.tree.TreeNodeUI = function(node){
35549     this.node = node;
35550     this.rendered = false;
35551     this.animating = false;
35552     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35553 };
35554
35555 Roo.tree.TreeNodeUI.prototype = {
35556     removeChild : function(node){
35557         if(this.rendered){
35558             this.ctNode.removeChild(node.ui.getEl());
35559         }
35560     },
35561
35562     beforeLoad : function(){
35563          this.addClass("x-tree-node-loading");
35564     },
35565
35566     afterLoad : function(){
35567          this.removeClass("x-tree-node-loading");
35568     },
35569
35570     onTextChange : function(node, text, oldText){
35571         if(this.rendered){
35572             this.textNode.innerHTML = text;
35573         }
35574     },
35575
35576     onDisableChange : function(node, state){
35577         this.disabled = state;
35578         if(state){
35579             this.addClass("x-tree-node-disabled");
35580         }else{
35581             this.removeClass("x-tree-node-disabled");
35582         }
35583     },
35584
35585     onSelectedChange : function(state){
35586         if(state){
35587             this.focus();
35588             this.addClass("x-tree-selected");
35589         }else{
35590             //this.blur();
35591             this.removeClass("x-tree-selected");
35592         }
35593     },
35594
35595     onMove : function(tree, node, oldParent, newParent, index, refNode){
35596         this.childIndent = null;
35597         if(this.rendered){
35598             var targetNode = newParent.ui.getContainer();
35599             if(!targetNode){//target not rendered
35600                 this.holder = document.createElement("div");
35601                 this.holder.appendChild(this.wrap);
35602                 return;
35603             }
35604             var insertBefore = refNode ? refNode.ui.getEl() : null;
35605             if(insertBefore){
35606                 targetNode.insertBefore(this.wrap, insertBefore);
35607             }else{
35608                 targetNode.appendChild(this.wrap);
35609             }
35610             this.node.renderIndent(true);
35611         }
35612     },
35613
35614     addClass : function(cls){
35615         if(this.elNode){
35616             Roo.fly(this.elNode).addClass(cls);
35617         }
35618     },
35619
35620     removeClass : function(cls){
35621         if(this.elNode){
35622             Roo.fly(this.elNode).removeClass(cls);
35623         }
35624     },
35625
35626     remove : function(){
35627         if(this.rendered){
35628             this.holder = document.createElement("div");
35629             this.holder.appendChild(this.wrap);
35630         }
35631     },
35632
35633     fireEvent : function(){
35634         return this.node.fireEvent.apply(this.node, arguments);
35635     },
35636
35637     initEvents : function(){
35638         this.node.on("move", this.onMove, this);
35639         var E = Roo.EventManager;
35640         var a = this.anchor;
35641
35642         var el = Roo.fly(a, '_treeui');
35643
35644         if(Roo.isOpera){ // opera render bug ignores the CSS
35645             el.setStyle("text-decoration", "none");
35646         }
35647
35648         el.on("click", this.onClick, this);
35649         el.on("dblclick", this.onDblClick, this);
35650
35651         if(this.checkbox){
35652             Roo.EventManager.on(this.checkbox,
35653                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35654         }
35655
35656         el.on("contextmenu", this.onContextMenu, this);
35657
35658         var icon = Roo.fly(this.iconNode);
35659         icon.on("click", this.onClick, this);
35660         icon.on("dblclick", this.onDblClick, this);
35661         icon.on("contextmenu", this.onContextMenu, this);
35662         E.on(this.ecNode, "click", this.ecClick, this, true);
35663
35664         if(this.node.disabled){
35665             this.addClass("x-tree-node-disabled");
35666         }
35667         if(this.node.hidden){
35668             this.addClass("x-tree-node-disabled");
35669         }
35670         var ot = this.node.getOwnerTree();
35671         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35672         if(dd && (!this.node.isRoot || ot.rootVisible)){
35673             Roo.dd.Registry.register(this.elNode, {
35674                 node: this.node,
35675                 handles: this.getDDHandles(),
35676                 isHandle: false
35677             });
35678         }
35679     },
35680
35681     getDDHandles : function(){
35682         return [this.iconNode, this.textNode];
35683     },
35684
35685     hide : function(){
35686         if(this.rendered){
35687             this.wrap.style.display = "none";
35688         }
35689     },
35690
35691     show : function(){
35692         if(this.rendered){
35693             this.wrap.style.display = "";
35694         }
35695     },
35696
35697     onContextMenu : function(e){
35698         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35699             e.preventDefault();
35700             this.focus();
35701             this.fireEvent("contextmenu", this.node, e);
35702         }
35703     },
35704
35705     onClick : function(e){
35706         if(this.dropping){
35707             e.stopEvent();
35708             return;
35709         }
35710         if(this.fireEvent("beforeclick", this.node, e) !== false){
35711             if(!this.disabled && this.node.attributes.href){
35712                 this.fireEvent("click", this.node, e);
35713                 return;
35714             }
35715             e.preventDefault();
35716             if(this.disabled){
35717                 return;
35718             }
35719
35720             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35721                 this.node.toggle();
35722             }
35723
35724             this.fireEvent("click", this.node, e);
35725         }else{
35726             e.stopEvent();
35727         }
35728     },
35729
35730     onDblClick : function(e){
35731         e.preventDefault();
35732         if(this.disabled){
35733             return;
35734         }
35735         if(this.checkbox){
35736             this.toggleCheck();
35737         }
35738         if(!this.animating && this.node.hasChildNodes()){
35739             this.node.toggle();
35740         }
35741         this.fireEvent("dblclick", this.node, e);
35742     },
35743
35744     onCheckChange : function(){
35745         var checked = this.checkbox.checked;
35746         this.node.attributes.checked = checked;
35747         this.fireEvent('checkchange', this.node, checked);
35748     },
35749
35750     ecClick : function(e){
35751         if(!this.animating && this.node.hasChildNodes()){
35752             this.node.toggle();
35753         }
35754     },
35755
35756     startDrop : function(){
35757         this.dropping = true;
35758     },
35759
35760     // delayed drop so the click event doesn't get fired on a drop
35761     endDrop : function(){
35762        setTimeout(function(){
35763            this.dropping = false;
35764        }.createDelegate(this), 50);
35765     },
35766
35767     expand : function(){
35768         this.updateExpandIcon();
35769         this.ctNode.style.display = "";
35770     },
35771
35772     focus : function(){
35773         if(!this.node.preventHScroll){
35774             try{this.anchor.focus();
35775             }catch(e){}
35776         }else if(!Roo.isIE){
35777             try{
35778                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35779                 var l = noscroll.scrollLeft;
35780                 this.anchor.focus();
35781                 noscroll.scrollLeft = l;
35782             }catch(e){}
35783         }
35784     },
35785
35786     toggleCheck : function(value){
35787         var cb = this.checkbox;
35788         if(cb){
35789             cb.checked = (value === undefined ? !cb.checked : value);
35790         }
35791     },
35792
35793     blur : function(){
35794         try{
35795             this.anchor.blur();
35796         }catch(e){}
35797     },
35798
35799     animExpand : function(callback){
35800         var ct = Roo.get(this.ctNode);
35801         ct.stopFx();
35802         if(!this.node.hasChildNodes()){
35803             this.updateExpandIcon();
35804             this.ctNode.style.display = "";
35805             Roo.callback(callback);
35806             return;
35807         }
35808         this.animating = true;
35809         this.updateExpandIcon();
35810
35811         ct.slideIn('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     highlight : function(){
35822         var tree = this.node.getOwnerTree();
35823         Roo.fly(this.wrap).highlight(
35824             tree.hlColor || "C3DAF9",
35825             {endColor: tree.hlBaseColor}
35826         );
35827     },
35828
35829     collapse : function(){
35830         this.updateExpandIcon();
35831         this.ctNode.style.display = "none";
35832     },
35833
35834     animCollapse : function(callback){
35835         var ct = Roo.get(this.ctNode);
35836         ct.enableDisplayMode('block');
35837         ct.stopFx();
35838
35839         this.animating = true;
35840         this.updateExpandIcon();
35841
35842         ct.slideOut('t', {
35843             callback : function(){
35844                this.animating = false;
35845                Roo.callback(callback);
35846             },
35847             scope: this,
35848             duration: this.node.ownerTree.duration || .25
35849         });
35850     },
35851
35852     getContainer : function(){
35853         return this.ctNode;
35854     },
35855
35856     getEl : function(){
35857         return this.wrap;
35858     },
35859
35860     appendDDGhost : function(ghostNode){
35861         ghostNode.appendChild(this.elNode.cloneNode(true));
35862     },
35863
35864     getDDRepairXY : function(){
35865         return Roo.lib.Dom.getXY(this.iconNode);
35866     },
35867
35868     onRender : function(){
35869         this.render();
35870     },
35871
35872     render : function(bulkRender){
35873         var n = this.node, a = n.attributes;
35874         var targetNode = n.parentNode ?
35875               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35876
35877         if(!this.rendered){
35878             this.rendered = true;
35879
35880             this.renderElements(n, a, targetNode, bulkRender);
35881
35882             if(a.qtip){
35883                if(this.textNode.setAttributeNS){
35884                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35885                    if(a.qtipTitle){
35886                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35887                    }
35888                }else{
35889                    this.textNode.setAttribute("ext:qtip", a.qtip);
35890                    if(a.qtipTitle){
35891                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35892                    }
35893                }
35894             }else if(a.qtipCfg){
35895                 a.qtipCfg.target = Roo.id(this.textNode);
35896                 Roo.QuickTips.register(a.qtipCfg);
35897             }
35898             this.initEvents();
35899             if(!this.node.expanded){
35900                 this.updateExpandIcon();
35901             }
35902         }else{
35903             if(bulkRender === true) {
35904                 targetNode.appendChild(this.wrap);
35905             }
35906         }
35907     },
35908
35909     renderElements : function(n, a, targetNode, bulkRender)
35910     {
35911         // add some indent caching, this helps performance when rendering a large tree
35912         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35913         var t = n.getOwnerTree();
35914         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35915         if (typeof(n.attributes.html) != 'undefined') {
35916             txt = n.attributes.html;
35917         }
35918         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35919         var cb = typeof a.checked == 'boolean';
35920         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35921         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35922             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35923             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35924             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35925             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35926             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35927              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35928                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35929             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35930             "</li>"];
35931
35932         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35933             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35934                                 n.nextSibling.ui.getEl(), buf.join(""));
35935         }else{
35936             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35937         }
35938
35939         this.elNode = this.wrap.childNodes[0];
35940         this.ctNode = this.wrap.childNodes[1];
35941         var cs = this.elNode.childNodes;
35942         this.indentNode = cs[0];
35943         this.ecNode = cs[1];
35944         this.iconNode = cs[2];
35945         var index = 3;
35946         if(cb){
35947             this.checkbox = cs[3];
35948             index++;
35949         }
35950         this.anchor = cs[index];
35951         this.textNode = cs[index].firstChild;
35952     },
35953
35954     getAnchor : function(){
35955         return this.anchor;
35956     },
35957
35958     getTextEl : function(){
35959         return this.textNode;
35960     },
35961
35962     getIconEl : function(){
35963         return this.iconNode;
35964     },
35965
35966     isChecked : function(){
35967         return this.checkbox ? this.checkbox.checked : false;
35968     },
35969
35970     updateExpandIcon : function(){
35971         if(this.rendered){
35972             var n = this.node, c1, c2;
35973             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35974             var hasChild = n.hasChildNodes();
35975             if(hasChild){
35976                 if(n.expanded){
35977                     cls += "-minus";
35978                     c1 = "x-tree-node-collapsed";
35979                     c2 = "x-tree-node-expanded";
35980                 }else{
35981                     cls += "-plus";
35982                     c1 = "x-tree-node-expanded";
35983                     c2 = "x-tree-node-collapsed";
35984                 }
35985                 if(this.wasLeaf){
35986                     this.removeClass("x-tree-node-leaf");
35987                     this.wasLeaf = false;
35988                 }
35989                 if(this.c1 != c1 || this.c2 != c2){
35990                     Roo.fly(this.elNode).replaceClass(c1, c2);
35991                     this.c1 = c1; this.c2 = c2;
35992                 }
35993             }else{
35994                 // this changes non-leafs into leafs if they have no children.
35995                 // it's not very rational behaviour..
35996                 
35997                 if(!this.wasLeaf && this.node.leaf){
35998                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35999                     delete this.c1;
36000                     delete this.c2;
36001                     this.wasLeaf = true;
36002                 }
36003             }
36004             var ecc = "x-tree-ec-icon "+cls;
36005             if(this.ecc != ecc){
36006                 this.ecNode.className = ecc;
36007                 this.ecc = ecc;
36008             }
36009         }
36010     },
36011
36012     getChildIndent : function(){
36013         if(!this.childIndent){
36014             var buf = [];
36015             var p = this.node;
36016             while(p){
36017                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
36018                     if(!p.isLast()) {
36019                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
36020                     } else {
36021                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
36022                     }
36023                 }
36024                 p = p.parentNode;
36025             }
36026             this.childIndent = buf.join("");
36027         }
36028         return this.childIndent;
36029     },
36030
36031     renderIndent : function(){
36032         if(this.rendered){
36033             var indent = "";
36034             var p = this.node.parentNode;
36035             if(p){
36036                 indent = p.ui.getChildIndent();
36037             }
36038             if(this.indentMarkup != indent){ // don't rerender if not required
36039                 this.indentNode.innerHTML = indent;
36040                 this.indentMarkup = indent;
36041             }
36042             this.updateExpandIcon();
36043         }
36044     }
36045 };
36046
36047 Roo.tree.RootTreeNodeUI = function(){
36048     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36049 };
36050 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36051     render : function(){
36052         if(!this.rendered){
36053             var targetNode = this.node.ownerTree.innerCt.dom;
36054             this.node.expanded = true;
36055             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36056             this.wrap = this.ctNode = targetNode.firstChild;
36057         }
36058     },
36059     collapse : function(){
36060     },
36061     expand : function(){
36062     }
36063 });/*
36064  * Based on:
36065  * Ext JS Library 1.1.1
36066  * Copyright(c) 2006-2007, Ext JS, LLC.
36067  *
36068  * Originally Released Under LGPL - original licence link has changed is not relivant.
36069  *
36070  * Fork - LGPL
36071  * <script type="text/javascript">
36072  */
36073 /**
36074  * @class Roo.tree.TreeLoader
36075  * @extends Roo.util.Observable
36076  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36077  * nodes from a specified URL. The response must be a javascript Array definition
36078  * who's elements are node definition objects. eg:
36079  * <pre><code>
36080 {  success : true,
36081    data :      [
36082    
36083     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36084     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36085     ]
36086 }
36087
36088
36089 </code></pre>
36090  * <br><br>
36091  * The old style respose with just an array is still supported, but not recommended.
36092  * <br><br>
36093  *
36094  * A server request is sent, and child nodes are loaded only when a node is expanded.
36095  * The loading node's id is passed to the server under the parameter name "node" to
36096  * enable the server to produce the correct child nodes.
36097  * <br><br>
36098  * To pass extra parameters, an event handler may be attached to the "beforeload"
36099  * event, and the parameters specified in the TreeLoader's baseParams property:
36100  * <pre><code>
36101     myTreeLoader.on("beforeload", function(treeLoader, node) {
36102         this.baseParams.category = node.attributes.category;
36103     }, this);
36104     
36105 </code></pre>
36106  *
36107  * This would pass an HTTP parameter called "category" to the server containing
36108  * the value of the Node's "category" attribute.
36109  * @constructor
36110  * Creates a new Treeloader.
36111  * @param {Object} config A config object containing config properties.
36112  */
36113 Roo.tree.TreeLoader = function(config){
36114     this.baseParams = {};
36115     this.requestMethod = "POST";
36116     Roo.apply(this, config);
36117
36118     this.addEvents({
36119     
36120         /**
36121          * @event beforeload
36122          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36123          * @param {Object} This TreeLoader object.
36124          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36125          * @param {Object} callback The callback function specified in the {@link #load} call.
36126          */
36127         beforeload : true,
36128         /**
36129          * @event load
36130          * Fires when the node has been successfuly loaded.
36131          * @param {Object} This TreeLoader object.
36132          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36133          * @param {Object} response The response object containing the data from the server.
36134          */
36135         load : true,
36136         /**
36137          * @event loadexception
36138          * Fires if the network request failed.
36139          * @param {Object} This TreeLoader object.
36140          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36141          * @param {Object} response The response object containing the data from the server.
36142          */
36143         loadexception : true,
36144         /**
36145          * @event create
36146          * Fires before a node is created, enabling you to return custom Node types 
36147          * @param {Object} This TreeLoader object.
36148          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36149          */
36150         create : true
36151     });
36152
36153     Roo.tree.TreeLoader.superclass.constructor.call(this);
36154 };
36155
36156 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36157     /**
36158     * @cfg {String} dataUrl The URL from which to request a Json string which
36159     * specifies an array of node definition object representing the child nodes
36160     * to be loaded.
36161     */
36162     /**
36163     * @cfg {String} requestMethod either GET or POST
36164     * defaults to POST (due to BC)
36165     * to be loaded.
36166     */
36167     /**
36168     * @cfg {Object} baseParams (optional) An object containing properties which
36169     * specify HTTP parameters to be passed to each request for child nodes.
36170     */
36171     /**
36172     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36173     * created by this loader. If the attributes sent by the server have an attribute in this object,
36174     * they take priority.
36175     */
36176     /**
36177     * @cfg {Object} uiProviders (optional) An object containing properties which
36178     * 
36179     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36180     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36181     * <i>uiProvider</i> attribute of a returned child node is a string rather
36182     * than a reference to a TreeNodeUI implementation, this that string value
36183     * is used as a property name in the uiProviders object. You can define the provider named
36184     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36185     */
36186     uiProviders : {},
36187
36188     /**
36189     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36190     * child nodes before loading.
36191     */
36192     clearOnLoad : true,
36193
36194     /**
36195     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36196     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36197     * Grid query { data : [ .....] }
36198     */
36199     
36200     root : false,
36201      /**
36202     * @cfg {String} queryParam (optional) 
36203     * Name of the query as it will be passed on the querystring (defaults to 'node')
36204     * eg. the request will be ?node=[id]
36205     */
36206     
36207     
36208     queryParam: false,
36209     
36210     /**
36211      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36212      * This is called automatically when a node is expanded, but may be used to reload
36213      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36214      * @param {Roo.tree.TreeNode} node
36215      * @param {Function} callback
36216      */
36217     load : function(node, callback){
36218         if(this.clearOnLoad){
36219             while(node.firstChild){
36220                 node.removeChild(node.firstChild);
36221             }
36222         }
36223         if(node.attributes.children){ // preloaded json children
36224             var cs = node.attributes.children;
36225             for(var i = 0, len = cs.length; i < len; i++){
36226                 node.appendChild(this.createNode(cs[i]));
36227             }
36228             if(typeof callback == "function"){
36229                 callback();
36230             }
36231         }else if(this.dataUrl){
36232             this.requestData(node, callback);
36233         }
36234     },
36235
36236     getParams: function(node){
36237         var buf = [], bp = this.baseParams;
36238         for(var key in bp){
36239             if(typeof bp[key] != "function"){
36240                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36241             }
36242         }
36243         var n = this.queryParam === false ? 'node' : this.queryParam;
36244         buf.push(n + "=", encodeURIComponent(node.id));
36245         return buf.join("");
36246     },
36247
36248     requestData : function(node, callback){
36249         if(this.fireEvent("beforeload", this, node, callback) !== false){
36250             this.transId = Roo.Ajax.request({
36251                 method:this.requestMethod,
36252                 url: this.dataUrl||this.url,
36253                 success: this.handleResponse,
36254                 failure: this.handleFailure,
36255                 scope: this,
36256                 argument: {callback: callback, node: node},
36257                 params: this.getParams(node)
36258             });
36259         }else{
36260             // if the load is cancelled, make sure we notify
36261             // the node that we are done
36262             if(typeof callback == "function"){
36263                 callback();
36264             }
36265         }
36266     },
36267
36268     isLoading : function(){
36269         return this.transId ? true : false;
36270     },
36271
36272     abort : function(){
36273         if(this.isLoading()){
36274             Roo.Ajax.abort(this.transId);
36275         }
36276     },
36277
36278     // private
36279     createNode : function(attr)
36280     {
36281         // apply baseAttrs, nice idea Corey!
36282         if(this.baseAttrs){
36283             Roo.applyIf(attr, this.baseAttrs);
36284         }
36285         if(this.applyLoader !== false){
36286             attr.loader = this;
36287         }
36288         // uiProvider = depreciated..
36289         
36290         if(typeof(attr.uiProvider) == 'string'){
36291            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36292                 /**  eval:var:attr */ eval(attr.uiProvider);
36293         }
36294         if(typeof(this.uiProviders['default']) != 'undefined') {
36295             attr.uiProvider = this.uiProviders['default'];
36296         }
36297         
36298         this.fireEvent('create', this, attr);
36299         
36300         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36301         return(attr.leaf ?
36302                         new Roo.tree.TreeNode(attr) :
36303                         new Roo.tree.AsyncTreeNode(attr));
36304     },
36305
36306     processResponse : function(response, node, callback)
36307     {
36308         var json = response.responseText;
36309         try {
36310             
36311             var o = Roo.decode(json);
36312             
36313             if (this.root === false && typeof(o.success) != undefined) {
36314                 this.root = 'data'; // the default behaviour for list like data..
36315                 }
36316                 
36317             if (this.root !== false &&  !o.success) {
36318                 // it's a failure condition.
36319                 var a = response.argument;
36320                 this.fireEvent("loadexception", this, a.node, response);
36321                 Roo.log("Load failed - should have a handler really");
36322                 return;
36323             }
36324             
36325             
36326             
36327             if (this.root !== false) {
36328                  o = o[this.root];
36329             }
36330             
36331             for(var i = 0, len = o.length; i < len; i++){
36332                 var n = this.createNode(o[i]);
36333                 if(n){
36334                     node.appendChild(n);
36335                 }
36336             }
36337             if(typeof callback == "function"){
36338                 callback(this, node);
36339             }
36340         }catch(e){
36341             this.handleFailure(response);
36342         }
36343     },
36344
36345     handleResponse : function(response){
36346         this.transId = false;
36347         var a = response.argument;
36348         this.processResponse(response, a.node, a.callback);
36349         this.fireEvent("load", this, a.node, response);
36350     },
36351
36352     handleFailure : function(response)
36353     {
36354         // should handle failure better..
36355         this.transId = false;
36356         var a = response.argument;
36357         this.fireEvent("loadexception", this, a.node, response);
36358         if(typeof a.callback == "function"){
36359             a.callback(this, a.node);
36360         }
36361     }
36362 });/*
36363  * Based on:
36364  * Ext JS Library 1.1.1
36365  * Copyright(c) 2006-2007, Ext JS, LLC.
36366  *
36367  * Originally Released Under LGPL - original licence link has changed is not relivant.
36368  *
36369  * Fork - LGPL
36370  * <script type="text/javascript">
36371  */
36372
36373 /**
36374 * @class Roo.tree.TreeFilter
36375 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36376 * @param {TreePanel} tree
36377 * @param {Object} config (optional)
36378  */
36379 Roo.tree.TreeFilter = function(tree, config){
36380     this.tree = tree;
36381     this.filtered = {};
36382     Roo.apply(this, config);
36383 };
36384
36385 Roo.tree.TreeFilter.prototype = {
36386     clearBlank:false,
36387     reverse:false,
36388     autoClear:false,
36389     remove:false,
36390
36391      /**
36392      * Filter the data by a specific attribute.
36393      * @param {String/RegExp} value Either string that the attribute value
36394      * should start with or a RegExp to test against the attribute
36395      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36396      * @param {TreeNode} startNode (optional) The node to start the filter at.
36397      */
36398     filter : function(value, attr, startNode){
36399         attr = attr || "text";
36400         var f;
36401         if(typeof value == "string"){
36402             var vlen = value.length;
36403             // auto clear empty filter
36404             if(vlen == 0 && this.clearBlank){
36405                 this.clear();
36406                 return;
36407             }
36408             value = value.toLowerCase();
36409             f = function(n){
36410                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36411             };
36412         }else if(value.exec){ // regex?
36413             f = function(n){
36414                 return value.test(n.attributes[attr]);
36415             };
36416         }else{
36417             throw 'Illegal filter type, must be string or regex';
36418         }
36419         this.filterBy(f, null, startNode);
36420         },
36421
36422     /**
36423      * Filter by a function. The passed function will be called with each
36424      * node in the tree (or from the startNode). If the function returns true, the node is kept
36425      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36426      * @param {Function} fn The filter function
36427      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36428      */
36429     filterBy : function(fn, scope, startNode){
36430         startNode = startNode || this.tree.root;
36431         if(this.autoClear){
36432             this.clear();
36433         }
36434         var af = this.filtered, rv = this.reverse;
36435         var f = function(n){
36436             if(n == startNode){
36437                 return true;
36438             }
36439             if(af[n.id]){
36440                 return false;
36441             }
36442             var m = fn.call(scope || n, n);
36443             if(!m || rv){
36444                 af[n.id] = n;
36445                 n.ui.hide();
36446                 return false;
36447             }
36448             return true;
36449         };
36450         startNode.cascade(f);
36451         if(this.remove){
36452            for(var id in af){
36453                if(typeof id != "function"){
36454                    var n = af[id];
36455                    if(n && n.parentNode){
36456                        n.parentNode.removeChild(n);
36457                    }
36458                }
36459            }
36460         }
36461     },
36462
36463     /**
36464      * Clears the current filter. Note: with the "remove" option
36465      * set a filter cannot be cleared.
36466      */
36467     clear : function(){
36468         var t = this.tree;
36469         var af = this.filtered;
36470         for(var id in af){
36471             if(typeof id != "function"){
36472                 var n = af[id];
36473                 if(n){
36474                     n.ui.show();
36475                 }
36476             }
36477         }
36478         this.filtered = {};
36479     }
36480 };
36481 /*
36482  * Based on:
36483  * Ext JS Library 1.1.1
36484  * Copyright(c) 2006-2007, Ext JS, LLC.
36485  *
36486  * Originally Released Under LGPL - original licence link has changed is not relivant.
36487  *
36488  * Fork - LGPL
36489  * <script type="text/javascript">
36490  */
36491  
36492
36493 /**
36494  * @class Roo.tree.TreeSorter
36495  * Provides sorting of nodes in a TreePanel
36496  * 
36497  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36498  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36499  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36500  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36501  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36502  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36503  * @constructor
36504  * @param {TreePanel} tree
36505  * @param {Object} config
36506  */
36507 Roo.tree.TreeSorter = function(tree, config){
36508     Roo.apply(this, config);
36509     tree.on("beforechildrenrendered", this.doSort, this);
36510     tree.on("append", this.updateSort, this);
36511     tree.on("insert", this.updateSort, this);
36512     
36513     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36514     var p = this.property || "text";
36515     var sortType = this.sortType;
36516     var fs = this.folderSort;
36517     var cs = this.caseSensitive === true;
36518     var leafAttr = this.leafAttr || 'leaf';
36519
36520     this.sortFn = function(n1, n2){
36521         if(fs){
36522             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36523                 return 1;
36524             }
36525             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36526                 return -1;
36527             }
36528         }
36529         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36530         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36531         if(v1 < v2){
36532                         return dsc ? +1 : -1;
36533                 }else if(v1 > v2){
36534                         return dsc ? -1 : +1;
36535         }else{
36536                 return 0;
36537         }
36538     };
36539 };
36540
36541 Roo.tree.TreeSorter.prototype = {
36542     doSort : function(node){
36543         node.sort(this.sortFn);
36544     },
36545     
36546     compareNodes : function(n1, n2){
36547         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36548     },
36549     
36550     updateSort : function(tree, node){
36551         if(node.childrenRendered){
36552             this.doSort.defer(1, this, [node]);
36553         }
36554     }
36555 };/*
36556  * Based on:
36557  * Ext JS Library 1.1.1
36558  * Copyright(c) 2006-2007, Ext JS, LLC.
36559  *
36560  * Originally Released Under LGPL - original licence link has changed is not relivant.
36561  *
36562  * Fork - LGPL
36563  * <script type="text/javascript">
36564  */
36565
36566 if(Roo.dd.DropZone){
36567     
36568 Roo.tree.TreeDropZone = function(tree, config){
36569     this.allowParentInsert = false;
36570     this.allowContainerDrop = false;
36571     this.appendOnly = false;
36572     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36573     this.tree = tree;
36574     this.lastInsertClass = "x-tree-no-status";
36575     this.dragOverData = {};
36576 };
36577
36578 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36579     ddGroup : "TreeDD",
36580     scroll:  true,
36581     
36582     expandDelay : 1000,
36583     
36584     expandNode : function(node){
36585         if(node.hasChildNodes() && !node.isExpanded()){
36586             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36587         }
36588     },
36589     
36590     queueExpand : function(node){
36591         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36592     },
36593     
36594     cancelExpand : function(){
36595         if(this.expandProcId){
36596             clearTimeout(this.expandProcId);
36597             this.expandProcId = false;
36598         }
36599     },
36600     
36601     isValidDropPoint : function(n, pt, dd, e, data){
36602         if(!n || !data){ return false; }
36603         var targetNode = n.node;
36604         var dropNode = data.node;
36605         // default drop rules
36606         if(!(targetNode && targetNode.isTarget && pt)){
36607             return false;
36608         }
36609         if(pt == "append" && targetNode.allowChildren === false){
36610             return false;
36611         }
36612         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36613             return false;
36614         }
36615         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36616             return false;
36617         }
36618         // reuse the object
36619         var overEvent = this.dragOverData;
36620         overEvent.tree = this.tree;
36621         overEvent.target = targetNode;
36622         overEvent.data = data;
36623         overEvent.point = pt;
36624         overEvent.source = dd;
36625         overEvent.rawEvent = e;
36626         overEvent.dropNode = dropNode;
36627         overEvent.cancel = false;  
36628         var result = this.tree.fireEvent("nodedragover", overEvent);
36629         return overEvent.cancel === false && result !== false;
36630     },
36631     
36632     getDropPoint : function(e, n, dd)
36633     {
36634         var tn = n.node;
36635         if(tn.isRoot){
36636             return tn.allowChildren !== false ? "append" : false; // always append for root
36637         }
36638         var dragEl = n.ddel;
36639         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36640         var y = Roo.lib.Event.getPageY(e);
36641         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36642         
36643         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36644         var noAppend = tn.allowChildren === false;
36645         if(this.appendOnly || tn.parentNode.allowChildren === false){
36646             return noAppend ? false : "append";
36647         }
36648         var noBelow = false;
36649         if(!this.allowParentInsert){
36650             noBelow = tn.hasChildNodes() && tn.isExpanded();
36651         }
36652         var q = (b - t) / (noAppend ? 2 : 3);
36653         if(y >= t && y < (t + q)){
36654             return "above";
36655         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36656             return "below";
36657         }else{
36658             return "append";
36659         }
36660     },
36661     
36662     onNodeEnter : function(n, dd, e, data)
36663     {
36664         this.cancelExpand();
36665     },
36666     
36667     onNodeOver : function(n, dd, e, data)
36668     {
36669        
36670         var pt = this.getDropPoint(e, n, dd);
36671         var node = n.node;
36672         
36673         // auto node expand check
36674         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36675             this.queueExpand(node);
36676         }else if(pt != "append"){
36677             this.cancelExpand();
36678         }
36679         
36680         // set the insert point style on the target node
36681         var returnCls = this.dropNotAllowed;
36682         if(this.isValidDropPoint(n, pt, dd, e, data)){
36683            if(pt){
36684                var el = n.ddel;
36685                var cls;
36686                if(pt == "above"){
36687                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36688                    cls = "x-tree-drag-insert-above";
36689                }else if(pt == "below"){
36690                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36691                    cls = "x-tree-drag-insert-below";
36692                }else{
36693                    returnCls = "x-tree-drop-ok-append";
36694                    cls = "x-tree-drag-append";
36695                }
36696                if(this.lastInsertClass != cls){
36697                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36698                    this.lastInsertClass = cls;
36699                }
36700            }
36701        }
36702        return returnCls;
36703     },
36704     
36705     onNodeOut : function(n, dd, e, data){
36706         
36707         this.cancelExpand();
36708         this.removeDropIndicators(n);
36709     },
36710     
36711     onNodeDrop : function(n, dd, e, data){
36712         var point = this.getDropPoint(e, n, dd);
36713         var targetNode = n.node;
36714         targetNode.ui.startDrop();
36715         if(!this.isValidDropPoint(n, point, dd, e, data)){
36716             targetNode.ui.endDrop();
36717             return false;
36718         }
36719         // first try to find the drop node
36720         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36721         var dropEvent = {
36722             tree : this.tree,
36723             target: targetNode,
36724             data: data,
36725             point: point,
36726             source: dd,
36727             rawEvent: e,
36728             dropNode: dropNode,
36729             cancel: !dropNode   
36730         };
36731         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36732         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36733             targetNode.ui.endDrop();
36734             return false;
36735         }
36736         // allow target changing
36737         targetNode = dropEvent.target;
36738         if(point == "append" && !targetNode.isExpanded()){
36739             targetNode.expand(false, null, function(){
36740                 this.completeDrop(dropEvent);
36741             }.createDelegate(this));
36742         }else{
36743             this.completeDrop(dropEvent);
36744         }
36745         return true;
36746     },
36747     
36748     completeDrop : function(de){
36749         var ns = de.dropNode, p = de.point, t = de.target;
36750         if(!(ns instanceof Array)){
36751             ns = [ns];
36752         }
36753         var n;
36754         for(var i = 0, len = ns.length; i < len; i++){
36755             n = ns[i];
36756             if(p == "above"){
36757                 t.parentNode.insertBefore(n, t);
36758             }else if(p == "below"){
36759                 t.parentNode.insertBefore(n, t.nextSibling);
36760             }else{
36761                 t.appendChild(n);
36762             }
36763         }
36764         n.ui.focus();
36765         if(this.tree.hlDrop){
36766             n.ui.highlight();
36767         }
36768         t.ui.endDrop();
36769         this.tree.fireEvent("nodedrop", de);
36770     },
36771     
36772     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36773         if(this.tree.hlDrop){
36774             dropNode.ui.focus();
36775             dropNode.ui.highlight();
36776         }
36777         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36778     },
36779     
36780     getTree : function(){
36781         return this.tree;
36782     },
36783     
36784     removeDropIndicators : function(n){
36785         if(n && n.ddel){
36786             var el = n.ddel;
36787             Roo.fly(el).removeClass([
36788                     "x-tree-drag-insert-above",
36789                     "x-tree-drag-insert-below",
36790                     "x-tree-drag-append"]);
36791             this.lastInsertClass = "_noclass";
36792         }
36793     },
36794     
36795     beforeDragDrop : function(target, e, id){
36796         this.cancelExpand();
36797         return true;
36798     },
36799     
36800     afterRepair : function(data){
36801         if(data && Roo.enableFx){
36802             data.node.ui.highlight();
36803         }
36804         this.hideProxy();
36805     } 
36806     
36807 });
36808
36809 }
36810 /*
36811  * Based on:
36812  * Ext JS Library 1.1.1
36813  * Copyright(c) 2006-2007, Ext JS, LLC.
36814  *
36815  * Originally Released Under LGPL - original licence link has changed is not relivant.
36816  *
36817  * Fork - LGPL
36818  * <script type="text/javascript">
36819  */
36820  
36821
36822 if(Roo.dd.DragZone){
36823 Roo.tree.TreeDragZone = function(tree, config){
36824     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36825     this.tree = tree;
36826 };
36827
36828 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36829     ddGroup : "TreeDD",
36830    
36831     onBeforeDrag : function(data, e){
36832         var n = data.node;
36833         return n && n.draggable && !n.disabled;
36834     },
36835      
36836     
36837     onInitDrag : function(e){
36838         var data = this.dragData;
36839         this.tree.getSelectionModel().select(data.node);
36840         this.proxy.update("");
36841         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36842         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36843     },
36844     
36845     getRepairXY : function(e, data){
36846         return data.node.ui.getDDRepairXY();
36847     },
36848     
36849     onEndDrag : function(data, e){
36850         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36851         
36852         
36853     },
36854     
36855     onValidDrop : function(dd, e, id){
36856         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36857         this.hideProxy();
36858     },
36859     
36860     beforeInvalidDrop : function(e, id){
36861         // this scrolls the original position back into view
36862         var sm = this.tree.getSelectionModel();
36863         sm.clearSelections();
36864         sm.select(this.dragData.node);
36865     }
36866 });
36867 }/*
36868  * Based on:
36869  * Ext JS Library 1.1.1
36870  * Copyright(c) 2006-2007, Ext JS, LLC.
36871  *
36872  * Originally Released Under LGPL - original licence link has changed is not relivant.
36873  *
36874  * Fork - LGPL
36875  * <script type="text/javascript">
36876  */
36877 /**
36878  * @class Roo.tree.TreeEditor
36879  * @extends Roo.Editor
36880  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36881  * as the editor field.
36882  * @constructor
36883  * @param {Object} config (used to be the tree panel.)
36884  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36885  * 
36886  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36887  * @cfg {Roo.form.TextField|Object} field The field configuration
36888  *
36889  * 
36890  */
36891 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36892     var tree = config;
36893     var field;
36894     if (oldconfig) { // old style..
36895         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36896     } else {
36897         // new style..
36898         tree = config.tree;
36899         config.field = config.field  || {};
36900         config.field.xtype = 'TextField';
36901         field = Roo.factory(config.field, Roo.form);
36902     }
36903     config = config || {};
36904     
36905     
36906     this.addEvents({
36907         /**
36908          * @event beforenodeedit
36909          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36910          * false from the handler of this event.
36911          * @param {Editor} this
36912          * @param {Roo.tree.Node} node 
36913          */
36914         "beforenodeedit" : true
36915     });
36916     
36917     //Roo.log(config);
36918     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36919
36920     this.tree = tree;
36921
36922     tree.on('beforeclick', this.beforeNodeClick, this);
36923     tree.getTreeEl().on('mousedown', this.hide, this);
36924     this.on('complete', this.updateNode, this);
36925     this.on('beforestartedit', this.fitToTree, this);
36926     this.on('startedit', this.bindScroll, this, {delay:10});
36927     this.on('specialkey', this.onSpecialKey, this);
36928 };
36929
36930 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36931     /**
36932      * @cfg {String} alignment
36933      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36934      */
36935     alignment: "l-l",
36936     // inherit
36937     autoSize: false,
36938     /**
36939      * @cfg {Boolean} hideEl
36940      * True to hide the bound element while the editor is displayed (defaults to false)
36941      */
36942     hideEl : false,
36943     /**
36944      * @cfg {String} cls
36945      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36946      */
36947     cls: "x-small-editor x-tree-editor",
36948     /**
36949      * @cfg {Boolean} shim
36950      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36951      */
36952     shim:false,
36953     // inherit
36954     shadow:"frame",
36955     /**
36956      * @cfg {Number} maxWidth
36957      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36958      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36959      * scroll and client offsets into account prior to each edit.
36960      */
36961     maxWidth: 250,
36962
36963     editDelay : 350,
36964
36965     // private
36966     fitToTree : function(ed, el){
36967         var td = this.tree.getTreeEl().dom, nd = el.dom;
36968         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36969             td.scrollLeft = nd.offsetLeft;
36970         }
36971         var w = Math.min(
36972                 this.maxWidth,
36973                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36974         this.setSize(w, '');
36975         
36976         return this.fireEvent('beforenodeedit', this, this.editNode);
36977         
36978     },
36979
36980     // private
36981     triggerEdit : function(node){
36982         this.completeEdit();
36983         this.editNode = node;
36984         this.startEdit(node.ui.textNode, node.text);
36985     },
36986
36987     // private
36988     bindScroll : function(){
36989         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36990     },
36991
36992     // private
36993     beforeNodeClick : function(node, e){
36994         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36995         this.lastClick = new Date();
36996         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36997             e.stopEvent();
36998             this.triggerEdit(node);
36999             return false;
37000         }
37001         return true;
37002     },
37003
37004     // private
37005     updateNode : function(ed, value){
37006         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
37007         this.editNode.setText(value);
37008     },
37009
37010     // private
37011     onHide : function(){
37012         Roo.tree.TreeEditor.superclass.onHide.call(this);
37013         if(this.editNode){
37014             this.editNode.ui.focus();
37015         }
37016     },
37017
37018     // private
37019     onSpecialKey : function(field, e){
37020         var k = e.getKey();
37021         if(k == e.ESC){
37022             e.stopEvent();
37023             this.cancelEdit();
37024         }else if(k == e.ENTER && !e.hasModifier()){
37025             e.stopEvent();
37026             this.completeEdit();
37027         }
37028     }
37029 });//<Script type="text/javascript">
37030 /*
37031  * Based on:
37032  * Ext JS Library 1.1.1
37033  * Copyright(c) 2006-2007, Ext JS, LLC.
37034  *
37035  * Originally Released Under LGPL - original licence link has changed is not relivant.
37036  *
37037  * Fork - LGPL
37038  * <script type="text/javascript">
37039  */
37040  
37041 /**
37042  * Not documented??? - probably should be...
37043  */
37044
37045 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37046     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37047     
37048     renderElements : function(n, a, targetNode, bulkRender){
37049         //consel.log("renderElements?");
37050         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37051
37052         var t = n.getOwnerTree();
37053         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37054         
37055         var cols = t.columns;
37056         var bw = t.borderWidth;
37057         var c = cols[0];
37058         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37059          var cb = typeof a.checked == "boolean";
37060         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37061         var colcls = 'x-t-' + tid + '-c0';
37062         var buf = [
37063             '<li class="x-tree-node">',
37064             
37065                 
37066                 '<div class="x-tree-node-el ', a.cls,'">',
37067                     // extran...
37068                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37069                 
37070                 
37071                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37072                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37073                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37074                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37075                            (a.iconCls ? ' '+a.iconCls : ''),
37076                            '" unselectable="on" />',
37077                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37078                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37079                              
37080                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37081                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37082                             '<span unselectable="on" qtip="' + tx + '">',
37083                              tx,
37084                              '</span></a>' ,
37085                     '</div>',
37086                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37087                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37088                  ];
37089         for(var i = 1, len = cols.length; i < len; i++){
37090             c = cols[i];
37091             colcls = 'x-t-' + tid + '-c' +i;
37092             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37093             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37094                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37095                       "</div>");
37096          }
37097          
37098          buf.push(
37099             '</a>',
37100             '<div class="x-clear"></div></div>',
37101             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37102             "</li>");
37103         
37104         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37105             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37106                                 n.nextSibling.ui.getEl(), buf.join(""));
37107         }else{
37108             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37109         }
37110         var el = this.wrap.firstChild;
37111         this.elRow = el;
37112         this.elNode = el.firstChild;
37113         this.ranchor = el.childNodes[1];
37114         this.ctNode = this.wrap.childNodes[1];
37115         var cs = el.firstChild.childNodes;
37116         this.indentNode = cs[0];
37117         this.ecNode = cs[1];
37118         this.iconNode = cs[2];
37119         var index = 3;
37120         if(cb){
37121             this.checkbox = cs[3];
37122             index++;
37123         }
37124         this.anchor = cs[index];
37125         
37126         this.textNode = cs[index].firstChild;
37127         
37128         //el.on("click", this.onClick, this);
37129         //el.on("dblclick", this.onDblClick, this);
37130         
37131         
37132        // console.log(this);
37133     },
37134     initEvents : function(){
37135         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37136         
37137             
37138         var a = this.ranchor;
37139
37140         var el = Roo.get(a);
37141
37142         if(Roo.isOpera){ // opera render bug ignores the CSS
37143             el.setStyle("text-decoration", "none");
37144         }
37145
37146         el.on("click", this.onClick, this);
37147         el.on("dblclick", this.onDblClick, this);
37148         el.on("contextmenu", this.onContextMenu, this);
37149         
37150     },
37151     
37152     /*onSelectedChange : function(state){
37153         if(state){
37154             this.focus();
37155             this.addClass("x-tree-selected");
37156         }else{
37157             //this.blur();
37158             this.removeClass("x-tree-selected");
37159         }
37160     },*/
37161     addClass : function(cls){
37162         if(this.elRow){
37163             Roo.fly(this.elRow).addClass(cls);
37164         }
37165         
37166     },
37167     
37168     
37169     removeClass : function(cls){
37170         if(this.elRow){
37171             Roo.fly(this.elRow).removeClass(cls);
37172         }
37173     }
37174
37175     
37176     
37177 });//<Script type="text/javascript">
37178
37179 /*
37180  * Based on:
37181  * Ext JS Library 1.1.1
37182  * Copyright(c) 2006-2007, Ext JS, LLC.
37183  *
37184  * Originally Released Under LGPL - original licence link has changed is not relivant.
37185  *
37186  * Fork - LGPL
37187  * <script type="text/javascript">
37188  */
37189  
37190
37191 /**
37192  * @class Roo.tree.ColumnTree
37193  * @extends Roo.data.TreePanel
37194  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37195  * @cfg {int} borderWidth  compined right/left border allowance
37196  * @constructor
37197  * @param {String/HTMLElement/Element} el The container element
37198  * @param {Object} config
37199  */
37200 Roo.tree.ColumnTree =  function(el, config)
37201 {
37202    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37203    this.addEvents({
37204         /**
37205         * @event resize
37206         * Fire this event on a container when it resizes
37207         * @param {int} w Width
37208         * @param {int} h Height
37209         */
37210        "resize" : true
37211     });
37212     this.on('resize', this.onResize, this);
37213 };
37214
37215 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37216     //lines:false,
37217     
37218     
37219     borderWidth: Roo.isBorderBox ? 0 : 2, 
37220     headEls : false,
37221     
37222     render : function(){
37223         // add the header.....
37224        
37225         Roo.tree.ColumnTree.superclass.render.apply(this);
37226         
37227         this.el.addClass('x-column-tree');
37228         
37229         this.headers = this.el.createChild(
37230             {cls:'x-tree-headers'},this.innerCt.dom);
37231    
37232         var cols = this.columns, c;
37233         var totalWidth = 0;
37234         this.headEls = [];
37235         var  len = cols.length;
37236         for(var i = 0; i < len; i++){
37237              c = cols[i];
37238              totalWidth += c.width;
37239             this.headEls.push(this.headers.createChild({
37240                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37241                  cn: {
37242                      cls:'x-tree-hd-text',
37243                      html: c.header
37244                  },
37245                  style:'width:'+(c.width-this.borderWidth)+'px;'
37246              }));
37247         }
37248         this.headers.createChild({cls:'x-clear'});
37249         // prevent floats from wrapping when clipped
37250         this.headers.setWidth(totalWidth);
37251         //this.innerCt.setWidth(totalWidth);
37252         this.innerCt.setStyle({ overflow: 'auto' });
37253         this.onResize(this.width, this.height);
37254              
37255         
37256     },
37257     onResize : function(w,h)
37258     {
37259         this.height = h;
37260         this.width = w;
37261         // resize cols..
37262         this.innerCt.setWidth(this.width);
37263         this.innerCt.setHeight(this.height-20);
37264         
37265         // headers...
37266         var cols = this.columns, c;
37267         var totalWidth = 0;
37268         var expEl = false;
37269         var len = cols.length;
37270         for(var i = 0; i < len; i++){
37271             c = cols[i];
37272             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37273                 // it's the expander..
37274                 expEl  = this.headEls[i];
37275                 continue;
37276             }
37277             totalWidth += c.width;
37278             
37279         }
37280         if (expEl) {
37281             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37282         }
37283         this.headers.setWidth(w-20);
37284
37285         
37286         
37287         
37288     }
37289 });
37290 /*
37291  * Based on:
37292  * Ext JS Library 1.1.1
37293  * Copyright(c) 2006-2007, Ext JS, LLC.
37294  *
37295  * Originally Released Under LGPL - original licence link has changed is not relivant.
37296  *
37297  * Fork - LGPL
37298  * <script type="text/javascript">
37299  */
37300  
37301 /**
37302  * @class Roo.menu.Menu
37303  * @extends Roo.util.Observable
37304  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37305  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37306  * @constructor
37307  * Creates a new Menu
37308  * @param {Object} config Configuration options
37309  */
37310 Roo.menu.Menu = function(config){
37311     
37312     Roo.menu.Menu.superclass.constructor.call(this, config);
37313     
37314     this.id = this.id || Roo.id();
37315     this.addEvents({
37316         /**
37317          * @event beforeshow
37318          * Fires before this menu is displayed
37319          * @param {Roo.menu.Menu} this
37320          */
37321         beforeshow : true,
37322         /**
37323          * @event beforehide
37324          * Fires before this menu is hidden
37325          * @param {Roo.menu.Menu} this
37326          */
37327         beforehide : true,
37328         /**
37329          * @event show
37330          * Fires after this menu is displayed
37331          * @param {Roo.menu.Menu} this
37332          */
37333         show : true,
37334         /**
37335          * @event hide
37336          * Fires after this menu is hidden
37337          * @param {Roo.menu.Menu} this
37338          */
37339         hide : true,
37340         /**
37341          * @event click
37342          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37343          * @param {Roo.menu.Menu} this
37344          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37345          * @param {Roo.EventObject} e
37346          */
37347         click : true,
37348         /**
37349          * @event mouseover
37350          * Fires when the mouse is hovering over this menu
37351          * @param {Roo.menu.Menu} this
37352          * @param {Roo.EventObject} e
37353          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37354          */
37355         mouseover : true,
37356         /**
37357          * @event mouseout
37358          * Fires when the mouse exits this menu
37359          * @param {Roo.menu.Menu} this
37360          * @param {Roo.EventObject} e
37361          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37362          */
37363         mouseout : true,
37364         /**
37365          * @event itemclick
37366          * Fires when a menu item contained in this menu is clicked
37367          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37368          * @param {Roo.EventObject} e
37369          */
37370         itemclick: true
37371     });
37372     if (this.registerMenu) {
37373         Roo.menu.MenuMgr.register(this);
37374     }
37375     
37376     var mis = this.items;
37377     this.items = new Roo.util.MixedCollection();
37378     if(mis){
37379         this.add.apply(this, mis);
37380     }
37381 };
37382
37383 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37384     /**
37385      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37386      */
37387     minWidth : 120,
37388     /**
37389      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37390      * for bottom-right shadow (defaults to "sides")
37391      */
37392     shadow : "sides",
37393     /**
37394      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37395      * this menu (defaults to "tl-tr?")
37396      */
37397     subMenuAlign : "tl-tr?",
37398     /**
37399      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37400      * relative to its element of origin (defaults to "tl-bl?")
37401      */
37402     defaultAlign : "tl-bl?",
37403     /**
37404      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37405      */
37406     allowOtherMenus : false,
37407     /**
37408      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37409      */
37410     registerMenu : true,
37411
37412     hidden:true,
37413
37414     // private
37415     render : function(){
37416         if(this.el){
37417             return;
37418         }
37419         var el = this.el = new Roo.Layer({
37420             cls: "x-menu",
37421             shadow:this.shadow,
37422             constrain: false,
37423             parentEl: this.parentEl || document.body,
37424             zindex:15000
37425         });
37426
37427         this.keyNav = new Roo.menu.MenuNav(this);
37428
37429         if(this.plain){
37430             el.addClass("x-menu-plain");
37431         }
37432         if(this.cls){
37433             el.addClass(this.cls);
37434         }
37435         // generic focus element
37436         this.focusEl = el.createChild({
37437             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37438         });
37439         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37440         //disabling touch- as it's causing issues ..
37441         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37442         ul.on('click'   , this.onClick, this);
37443         
37444         
37445         ul.on("mouseover", this.onMouseOver, this);
37446         ul.on("mouseout", this.onMouseOut, this);
37447         this.items.each(function(item){
37448             if (item.hidden) {
37449                 return;
37450             }
37451             
37452             var li = document.createElement("li");
37453             li.className = "x-menu-list-item";
37454             ul.dom.appendChild(li);
37455             item.render(li, this);
37456         }, this);
37457         this.ul = ul;
37458         this.autoWidth();
37459     },
37460
37461     // private
37462     autoWidth : function(){
37463         var el = this.el, ul = this.ul;
37464         if(!el){
37465             return;
37466         }
37467         var w = this.width;
37468         if(w){
37469             el.setWidth(w);
37470         }else if(Roo.isIE){
37471             el.setWidth(this.minWidth);
37472             var t = el.dom.offsetWidth; // force recalc
37473             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37474         }
37475     },
37476
37477     // private
37478     delayAutoWidth : function(){
37479         if(this.rendered){
37480             if(!this.awTask){
37481                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37482             }
37483             this.awTask.delay(20);
37484         }
37485     },
37486
37487     // private
37488     findTargetItem : function(e){
37489         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37490         if(t && t.menuItemId){
37491             return this.items.get(t.menuItemId);
37492         }
37493     },
37494
37495     // private
37496     onClick : function(e){
37497         Roo.log("menu.onClick");
37498         var t = this.findTargetItem(e);
37499         if(!t){
37500             return;
37501         }
37502         Roo.log(e);
37503         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37504             if(t == this.activeItem && t.shouldDeactivate(e)){
37505                 this.activeItem.deactivate();
37506                 delete this.activeItem;
37507                 return;
37508             }
37509             if(t.canActivate){
37510                 this.setActiveItem(t, true);
37511             }
37512             return;
37513             
37514             
37515         }
37516         
37517         t.onClick(e);
37518         this.fireEvent("click", this, t, e);
37519     },
37520
37521     // private
37522     setActiveItem : function(item, autoExpand){
37523         if(item != this.activeItem){
37524             if(this.activeItem){
37525                 this.activeItem.deactivate();
37526             }
37527             this.activeItem = item;
37528             item.activate(autoExpand);
37529         }else if(autoExpand){
37530             item.expandMenu();
37531         }
37532     },
37533
37534     // private
37535     tryActivate : function(start, step){
37536         var items = this.items;
37537         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37538             var item = items.get(i);
37539             if(!item.disabled && item.canActivate){
37540                 this.setActiveItem(item, false);
37541                 return item;
37542             }
37543         }
37544         return false;
37545     },
37546
37547     // private
37548     onMouseOver : function(e){
37549         var t;
37550         if(t = this.findTargetItem(e)){
37551             if(t.canActivate && !t.disabled){
37552                 this.setActiveItem(t, true);
37553             }
37554         }
37555         this.fireEvent("mouseover", this, e, t);
37556     },
37557
37558     // private
37559     onMouseOut : function(e){
37560         var t;
37561         if(t = this.findTargetItem(e)){
37562             if(t == this.activeItem && t.shouldDeactivate(e)){
37563                 this.activeItem.deactivate();
37564                 delete this.activeItem;
37565             }
37566         }
37567         this.fireEvent("mouseout", this, e, t);
37568     },
37569
37570     /**
37571      * Read-only.  Returns true if the menu is currently displayed, else false.
37572      * @type Boolean
37573      */
37574     isVisible : function(){
37575         return this.el && !this.hidden;
37576     },
37577
37578     /**
37579      * Displays this menu relative to another element
37580      * @param {String/HTMLElement/Roo.Element} element The element to align to
37581      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37582      * the element (defaults to this.defaultAlign)
37583      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37584      */
37585     show : function(el, pos, parentMenu){
37586         this.parentMenu = parentMenu;
37587         if(!this.el){
37588             this.render();
37589         }
37590         this.fireEvent("beforeshow", this);
37591         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37592     },
37593
37594     /**
37595      * Displays this menu at a specific xy position
37596      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37597      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37598      */
37599     showAt : function(xy, parentMenu, /* private: */_e){
37600         this.parentMenu = parentMenu;
37601         if(!this.el){
37602             this.render();
37603         }
37604         if(_e !== false){
37605             this.fireEvent("beforeshow", this);
37606             xy = this.el.adjustForConstraints(xy);
37607         }
37608         this.el.setXY(xy);
37609         this.el.show();
37610         this.hidden = false;
37611         this.focus();
37612         this.fireEvent("show", this);
37613     },
37614
37615     focus : function(){
37616         if(!this.hidden){
37617             this.doFocus.defer(50, this);
37618         }
37619     },
37620
37621     doFocus : function(){
37622         if(!this.hidden){
37623             this.focusEl.focus();
37624         }
37625     },
37626
37627     /**
37628      * Hides this menu and optionally all parent menus
37629      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37630      */
37631     hide : function(deep){
37632         if(this.el && this.isVisible()){
37633             this.fireEvent("beforehide", this);
37634             if(this.activeItem){
37635                 this.activeItem.deactivate();
37636                 this.activeItem = null;
37637             }
37638             this.el.hide();
37639             this.hidden = true;
37640             this.fireEvent("hide", this);
37641         }
37642         if(deep === true && this.parentMenu){
37643             this.parentMenu.hide(true);
37644         }
37645     },
37646
37647     /**
37648      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37649      * Any of the following are valid:
37650      * <ul>
37651      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37652      * <li>An HTMLElement object which will be converted to a menu item</li>
37653      * <li>A menu item config object that will be created as a new menu item</li>
37654      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37655      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37656      * </ul>
37657      * Usage:
37658      * <pre><code>
37659 // Create the menu
37660 var menu = new Roo.menu.Menu();
37661
37662 // Create a menu item to add by reference
37663 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37664
37665 // Add a bunch of items at once using different methods.
37666 // Only the last item added will be returned.
37667 var item = menu.add(
37668     menuItem,                // add existing item by ref
37669     'Dynamic Item',          // new TextItem
37670     '-',                     // new separator
37671     { text: 'Config Item' }  // new item by config
37672 );
37673 </code></pre>
37674      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37675      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37676      */
37677     add : function(){
37678         var a = arguments, l = a.length, item;
37679         for(var i = 0; i < l; i++){
37680             var el = a[i];
37681             if ((typeof(el) == "object") && el.xtype && el.xns) {
37682                 el = Roo.factory(el, Roo.menu);
37683             }
37684             
37685             if(el.render){ // some kind of Item
37686                 item = this.addItem(el);
37687             }else if(typeof el == "string"){ // string
37688                 if(el == "separator" || el == "-"){
37689                     item = this.addSeparator();
37690                 }else{
37691                     item = this.addText(el);
37692                 }
37693             }else if(el.tagName || el.el){ // element
37694                 item = this.addElement(el);
37695             }else if(typeof el == "object"){ // must be menu item config?
37696                 item = this.addMenuItem(el);
37697             }
37698         }
37699         return item;
37700     },
37701
37702     /**
37703      * Returns this menu's underlying {@link Roo.Element} object
37704      * @return {Roo.Element} The element
37705      */
37706     getEl : function(){
37707         if(!this.el){
37708             this.render();
37709         }
37710         return this.el;
37711     },
37712
37713     /**
37714      * Adds a separator bar to the menu
37715      * @return {Roo.menu.Item} The menu item that was added
37716      */
37717     addSeparator : function(){
37718         return this.addItem(new Roo.menu.Separator());
37719     },
37720
37721     /**
37722      * Adds an {@link Roo.Element} object to the menu
37723      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37724      * @return {Roo.menu.Item} The menu item that was added
37725      */
37726     addElement : function(el){
37727         return this.addItem(new Roo.menu.BaseItem(el));
37728     },
37729
37730     /**
37731      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37732      * @param {Roo.menu.Item} item The menu item to add
37733      * @return {Roo.menu.Item} The menu item that was added
37734      */
37735     addItem : function(item){
37736         this.items.add(item);
37737         if(this.ul){
37738             var li = document.createElement("li");
37739             li.className = "x-menu-list-item";
37740             this.ul.dom.appendChild(li);
37741             item.render(li, this);
37742             this.delayAutoWidth();
37743         }
37744         return item;
37745     },
37746
37747     /**
37748      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37749      * @param {Object} config A MenuItem config object
37750      * @return {Roo.menu.Item} The menu item that was added
37751      */
37752     addMenuItem : function(config){
37753         if(!(config instanceof Roo.menu.Item)){
37754             if(typeof config.checked == "boolean"){ // must be check menu item config?
37755                 config = new Roo.menu.CheckItem(config);
37756             }else{
37757                 config = new Roo.menu.Item(config);
37758             }
37759         }
37760         return this.addItem(config);
37761     },
37762
37763     /**
37764      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37765      * @param {String} text The text to display in the menu item
37766      * @return {Roo.menu.Item} The menu item that was added
37767      */
37768     addText : function(text){
37769         return this.addItem(new Roo.menu.TextItem({ text : text }));
37770     },
37771
37772     /**
37773      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37774      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37775      * @param {Roo.menu.Item} item The menu item to add
37776      * @return {Roo.menu.Item} The menu item that was added
37777      */
37778     insert : function(index, item){
37779         this.items.insert(index, item);
37780         if(this.ul){
37781             var li = document.createElement("li");
37782             li.className = "x-menu-list-item";
37783             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37784             item.render(li, this);
37785             this.delayAutoWidth();
37786         }
37787         return item;
37788     },
37789
37790     /**
37791      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37792      * @param {Roo.menu.Item} item The menu item to remove
37793      */
37794     remove : function(item){
37795         this.items.removeKey(item.id);
37796         item.destroy();
37797     },
37798
37799     /**
37800      * Removes and destroys all items in the menu
37801      */
37802     removeAll : function(){
37803         var f;
37804         while(f = this.items.first()){
37805             this.remove(f);
37806         }
37807     }
37808 });
37809
37810 // MenuNav is a private utility class used internally by the Menu
37811 Roo.menu.MenuNav = function(menu){
37812     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37813     this.scope = this.menu = menu;
37814 };
37815
37816 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37817     doRelay : function(e, h){
37818         var k = e.getKey();
37819         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37820             this.menu.tryActivate(0, 1);
37821             return false;
37822         }
37823         return h.call(this.scope || this, e, this.menu);
37824     },
37825
37826     up : function(e, m){
37827         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37828             m.tryActivate(m.items.length-1, -1);
37829         }
37830     },
37831
37832     down : function(e, m){
37833         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37834             m.tryActivate(0, 1);
37835         }
37836     },
37837
37838     right : function(e, m){
37839         if(m.activeItem){
37840             m.activeItem.expandMenu(true);
37841         }
37842     },
37843
37844     left : function(e, m){
37845         m.hide();
37846         if(m.parentMenu && m.parentMenu.activeItem){
37847             m.parentMenu.activeItem.activate();
37848         }
37849     },
37850
37851     enter : function(e, m){
37852         if(m.activeItem){
37853             e.stopPropagation();
37854             m.activeItem.onClick(e);
37855             m.fireEvent("click", this, m.activeItem);
37856             return true;
37857         }
37858     }
37859 });/*
37860  * Based on:
37861  * Ext JS Library 1.1.1
37862  * Copyright(c) 2006-2007, Ext JS, LLC.
37863  *
37864  * Originally Released Under LGPL - original licence link has changed is not relivant.
37865  *
37866  * Fork - LGPL
37867  * <script type="text/javascript">
37868  */
37869  
37870 /**
37871  * @class Roo.menu.MenuMgr
37872  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37873  * @singleton
37874  */
37875 Roo.menu.MenuMgr = function(){
37876    var menus, active, groups = {}, attached = false, lastShow = new Date();
37877
37878    // private - called when first menu is created
37879    function init(){
37880        menus = {};
37881        active = new Roo.util.MixedCollection();
37882        Roo.get(document).addKeyListener(27, function(){
37883            if(active.length > 0){
37884                hideAll();
37885            }
37886        });
37887    }
37888
37889    // private
37890    function hideAll(){
37891        if(active && active.length > 0){
37892            var c = active.clone();
37893            c.each(function(m){
37894                m.hide();
37895            });
37896        }
37897    }
37898
37899    // private
37900    function onHide(m){
37901        active.remove(m);
37902        if(active.length < 1){
37903            Roo.get(document).un("mousedown", onMouseDown);
37904            attached = false;
37905        }
37906    }
37907
37908    // private
37909    function onShow(m){
37910        var last = active.last();
37911        lastShow = new Date();
37912        active.add(m);
37913        if(!attached){
37914            Roo.get(document).on("mousedown", onMouseDown);
37915            attached = true;
37916        }
37917        if(m.parentMenu){
37918           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37919           m.parentMenu.activeChild = m;
37920        }else if(last && last.isVisible()){
37921           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37922        }
37923    }
37924
37925    // private
37926    function onBeforeHide(m){
37927        if(m.activeChild){
37928            m.activeChild.hide();
37929        }
37930        if(m.autoHideTimer){
37931            clearTimeout(m.autoHideTimer);
37932            delete m.autoHideTimer;
37933        }
37934    }
37935
37936    // private
37937    function onBeforeShow(m){
37938        var pm = m.parentMenu;
37939        if(!pm && !m.allowOtherMenus){
37940            hideAll();
37941        }else if(pm && pm.activeChild && active != m){
37942            pm.activeChild.hide();
37943        }
37944    }
37945
37946    // private
37947    function onMouseDown(e){
37948        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37949            hideAll();
37950        }
37951    }
37952
37953    // private
37954    function onBeforeCheck(mi, state){
37955        if(state){
37956            var g = groups[mi.group];
37957            for(var i = 0, l = g.length; i < l; i++){
37958                if(g[i] != mi){
37959                    g[i].setChecked(false);
37960                }
37961            }
37962        }
37963    }
37964
37965    return {
37966
37967        /**
37968         * Hides all menus that are currently visible
37969         */
37970        hideAll : function(){
37971             hideAll();  
37972        },
37973
37974        // private
37975        register : function(menu){
37976            if(!menus){
37977                init();
37978            }
37979            menus[menu.id] = menu;
37980            menu.on("beforehide", onBeforeHide);
37981            menu.on("hide", onHide);
37982            menu.on("beforeshow", onBeforeShow);
37983            menu.on("show", onShow);
37984            var g = menu.group;
37985            if(g && menu.events["checkchange"]){
37986                if(!groups[g]){
37987                    groups[g] = [];
37988                }
37989                groups[g].push(menu);
37990                menu.on("checkchange", onCheck);
37991            }
37992        },
37993
37994         /**
37995          * Returns a {@link Roo.menu.Menu} object
37996          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37997          * be used to generate and return a new Menu instance.
37998          */
37999        get : function(menu){
38000            if(typeof menu == "string"){ // menu id
38001                return menus[menu];
38002            }else if(menu.events){  // menu instance
38003                return menu;
38004            }else if(typeof menu.length == 'number'){ // array of menu items?
38005                return new Roo.menu.Menu({items:menu});
38006            }else{ // otherwise, must be a config
38007                return new Roo.menu.Menu(menu);
38008            }
38009        },
38010
38011        // private
38012        unregister : function(menu){
38013            delete menus[menu.id];
38014            menu.un("beforehide", onBeforeHide);
38015            menu.un("hide", onHide);
38016            menu.un("beforeshow", onBeforeShow);
38017            menu.un("show", onShow);
38018            var g = menu.group;
38019            if(g && menu.events["checkchange"]){
38020                groups[g].remove(menu);
38021                menu.un("checkchange", onCheck);
38022            }
38023        },
38024
38025        // private
38026        registerCheckable : function(menuItem){
38027            var g = menuItem.group;
38028            if(g){
38029                if(!groups[g]){
38030                    groups[g] = [];
38031                }
38032                groups[g].push(menuItem);
38033                menuItem.on("beforecheckchange", onBeforeCheck);
38034            }
38035        },
38036
38037        // private
38038        unregisterCheckable : function(menuItem){
38039            var g = menuItem.group;
38040            if(g){
38041                groups[g].remove(menuItem);
38042                menuItem.un("beforecheckchange", onBeforeCheck);
38043            }
38044        }
38045    };
38046 }();/*
38047  * Based on:
38048  * Ext JS Library 1.1.1
38049  * Copyright(c) 2006-2007, Ext JS, LLC.
38050  *
38051  * Originally Released Under LGPL - original licence link has changed is not relivant.
38052  *
38053  * Fork - LGPL
38054  * <script type="text/javascript">
38055  */
38056  
38057
38058 /**
38059  * @class Roo.menu.BaseItem
38060  * @extends Roo.Component
38061  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38062  * management and base configuration options shared by all menu components.
38063  * @constructor
38064  * Creates a new BaseItem
38065  * @param {Object} config Configuration options
38066  */
38067 Roo.menu.BaseItem = function(config){
38068     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38069
38070     this.addEvents({
38071         /**
38072          * @event click
38073          * Fires when this item is clicked
38074          * @param {Roo.menu.BaseItem} this
38075          * @param {Roo.EventObject} e
38076          */
38077         click: true,
38078         /**
38079          * @event activate
38080          * Fires when this item is activated
38081          * @param {Roo.menu.BaseItem} this
38082          */
38083         activate : true,
38084         /**
38085          * @event deactivate
38086          * Fires when this item is deactivated
38087          * @param {Roo.menu.BaseItem} this
38088          */
38089         deactivate : true
38090     });
38091
38092     if(this.handler){
38093         this.on("click", this.handler, this.scope, true);
38094     }
38095 };
38096
38097 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38098     /**
38099      * @cfg {Function} handler
38100      * A function that will handle the click event of this menu item (defaults to undefined)
38101      */
38102     /**
38103      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38104      */
38105     canActivate : false,
38106     
38107      /**
38108      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38109      */
38110     hidden: false,
38111     
38112     /**
38113      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38114      */
38115     activeClass : "x-menu-item-active",
38116     /**
38117      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38118      */
38119     hideOnClick : true,
38120     /**
38121      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38122      */
38123     hideDelay : 100,
38124
38125     // private
38126     ctype: "Roo.menu.BaseItem",
38127
38128     // private
38129     actionMode : "container",
38130
38131     // private
38132     render : function(container, parentMenu){
38133         this.parentMenu = parentMenu;
38134         Roo.menu.BaseItem.superclass.render.call(this, container);
38135         this.container.menuItemId = this.id;
38136     },
38137
38138     // private
38139     onRender : function(container, position){
38140         this.el = Roo.get(this.el);
38141         container.dom.appendChild(this.el.dom);
38142     },
38143
38144     // private
38145     onClick : function(e){
38146         if(!this.disabled && this.fireEvent("click", this, e) !== false
38147                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38148             this.handleClick(e);
38149         }else{
38150             e.stopEvent();
38151         }
38152     },
38153
38154     // private
38155     activate : function(){
38156         if(this.disabled){
38157             return false;
38158         }
38159         var li = this.container;
38160         li.addClass(this.activeClass);
38161         this.region = li.getRegion().adjust(2, 2, -2, -2);
38162         this.fireEvent("activate", this);
38163         return true;
38164     },
38165
38166     // private
38167     deactivate : function(){
38168         this.container.removeClass(this.activeClass);
38169         this.fireEvent("deactivate", this);
38170     },
38171
38172     // private
38173     shouldDeactivate : function(e){
38174         return !this.region || !this.region.contains(e.getPoint());
38175     },
38176
38177     // private
38178     handleClick : function(e){
38179         if(this.hideOnClick){
38180             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38181         }
38182     },
38183
38184     // private
38185     expandMenu : function(autoActivate){
38186         // do nothing
38187     },
38188
38189     // private
38190     hideMenu : function(){
38191         // do nothing
38192     }
38193 });/*
38194  * Based on:
38195  * Ext JS Library 1.1.1
38196  * Copyright(c) 2006-2007, Ext JS, LLC.
38197  *
38198  * Originally Released Under LGPL - original licence link has changed is not relivant.
38199  *
38200  * Fork - LGPL
38201  * <script type="text/javascript">
38202  */
38203  
38204 /**
38205  * @class Roo.menu.Adapter
38206  * @extends Roo.menu.BaseItem
38207  * 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.
38208  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38209  * @constructor
38210  * Creates a new Adapter
38211  * @param {Object} config Configuration options
38212  */
38213 Roo.menu.Adapter = function(component, config){
38214     Roo.menu.Adapter.superclass.constructor.call(this, config);
38215     this.component = component;
38216 };
38217 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38218     // private
38219     canActivate : true,
38220
38221     // private
38222     onRender : function(container, position){
38223         this.component.render(container);
38224         this.el = this.component.getEl();
38225     },
38226
38227     // private
38228     activate : function(){
38229         if(this.disabled){
38230             return false;
38231         }
38232         this.component.focus();
38233         this.fireEvent("activate", this);
38234         return true;
38235     },
38236
38237     // private
38238     deactivate : function(){
38239         this.fireEvent("deactivate", this);
38240     },
38241
38242     // private
38243     disable : function(){
38244         this.component.disable();
38245         Roo.menu.Adapter.superclass.disable.call(this);
38246     },
38247
38248     // private
38249     enable : function(){
38250         this.component.enable();
38251         Roo.menu.Adapter.superclass.enable.call(this);
38252     }
38253 });/*
38254  * Based on:
38255  * Ext JS Library 1.1.1
38256  * Copyright(c) 2006-2007, Ext JS, LLC.
38257  *
38258  * Originally Released Under LGPL - original licence link has changed is not relivant.
38259  *
38260  * Fork - LGPL
38261  * <script type="text/javascript">
38262  */
38263
38264 /**
38265  * @class Roo.menu.TextItem
38266  * @extends Roo.menu.BaseItem
38267  * Adds a static text string to a menu, usually used as either a heading or group separator.
38268  * Note: old style constructor with text is still supported.
38269  * 
38270  * @constructor
38271  * Creates a new TextItem
38272  * @param {Object} cfg Configuration
38273  */
38274 Roo.menu.TextItem = function(cfg){
38275     if (typeof(cfg) == 'string') {
38276         this.text = cfg;
38277     } else {
38278         Roo.apply(this,cfg);
38279     }
38280     
38281     Roo.menu.TextItem.superclass.constructor.call(this);
38282 };
38283
38284 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38285     /**
38286      * @cfg {Boolean} text Text to show on item.
38287      */
38288     text : '',
38289     
38290     /**
38291      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38292      */
38293     hideOnClick : false,
38294     /**
38295      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38296      */
38297     itemCls : "x-menu-text",
38298
38299     // private
38300     onRender : function(){
38301         var s = document.createElement("span");
38302         s.className = this.itemCls;
38303         s.innerHTML = this.text;
38304         this.el = s;
38305         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38306     }
38307 });/*
38308  * Based on:
38309  * Ext JS Library 1.1.1
38310  * Copyright(c) 2006-2007, Ext JS, LLC.
38311  *
38312  * Originally Released Under LGPL - original licence link has changed is not relivant.
38313  *
38314  * Fork - LGPL
38315  * <script type="text/javascript">
38316  */
38317
38318 /**
38319  * @class Roo.menu.Separator
38320  * @extends Roo.menu.BaseItem
38321  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38322  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38323  * @constructor
38324  * @param {Object} config Configuration options
38325  */
38326 Roo.menu.Separator = function(config){
38327     Roo.menu.Separator.superclass.constructor.call(this, config);
38328 };
38329
38330 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38331     /**
38332      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38333      */
38334     itemCls : "x-menu-sep",
38335     /**
38336      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38337      */
38338     hideOnClick : false,
38339
38340     // private
38341     onRender : function(li){
38342         var s = document.createElement("span");
38343         s.className = this.itemCls;
38344         s.innerHTML = "&#160;";
38345         this.el = s;
38346         li.addClass("x-menu-sep-li");
38347         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38348     }
38349 });/*
38350  * Based on:
38351  * Ext JS Library 1.1.1
38352  * Copyright(c) 2006-2007, Ext JS, LLC.
38353  *
38354  * Originally Released Under LGPL - original licence link has changed is not relivant.
38355  *
38356  * Fork - LGPL
38357  * <script type="text/javascript">
38358  */
38359 /**
38360  * @class Roo.menu.Item
38361  * @extends Roo.menu.BaseItem
38362  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38363  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38364  * activation and click handling.
38365  * @constructor
38366  * Creates a new Item
38367  * @param {Object} config Configuration options
38368  */
38369 Roo.menu.Item = function(config){
38370     Roo.menu.Item.superclass.constructor.call(this, config);
38371     if(this.menu){
38372         this.menu = Roo.menu.MenuMgr.get(this.menu);
38373     }
38374 };
38375 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38376     
38377     /**
38378      * @cfg {String} text
38379      * The text to show on the menu item.
38380      */
38381     text: '',
38382      /**
38383      * @cfg {String} HTML to render in menu
38384      * The text to show on the menu item (HTML version).
38385      */
38386     html: '',
38387     /**
38388      * @cfg {String} icon
38389      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38390      */
38391     icon: undefined,
38392     /**
38393      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38394      */
38395     itemCls : "x-menu-item",
38396     /**
38397      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38398      */
38399     canActivate : true,
38400     /**
38401      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38402      */
38403     showDelay: 200,
38404     // doc'd in BaseItem
38405     hideDelay: 200,
38406
38407     // private
38408     ctype: "Roo.menu.Item",
38409     
38410     // private
38411     onRender : function(container, position){
38412         var el = document.createElement("a");
38413         el.hideFocus = true;
38414         el.unselectable = "on";
38415         el.href = this.href || "#";
38416         if(this.hrefTarget){
38417             el.target = this.hrefTarget;
38418         }
38419         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38420         
38421         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38422         
38423         el.innerHTML = String.format(
38424                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38425                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38426         this.el = el;
38427         Roo.menu.Item.superclass.onRender.call(this, container, position);
38428     },
38429
38430     /**
38431      * Sets the text to display in this menu item
38432      * @param {String} text The text to display
38433      * @param {Boolean} isHTML true to indicate text is pure html.
38434      */
38435     setText : function(text, isHTML){
38436         if (isHTML) {
38437             this.html = text;
38438         } else {
38439             this.text = text;
38440             this.html = '';
38441         }
38442         if(this.rendered){
38443             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38444      
38445             this.el.update(String.format(
38446                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38447                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38448             this.parentMenu.autoWidth();
38449         }
38450     },
38451
38452     // private
38453     handleClick : function(e){
38454         if(!this.href){ // if no link defined, stop the event automatically
38455             e.stopEvent();
38456         }
38457         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38458     },
38459
38460     // private
38461     activate : function(autoExpand){
38462         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38463             this.focus();
38464             if(autoExpand){
38465                 this.expandMenu();
38466             }
38467         }
38468         return true;
38469     },
38470
38471     // private
38472     shouldDeactivate : function(e){
38473         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38474             if(this.menu && this.menu.isVisible()){
38475                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38476             }
38477             return true;
38478         }
38479         return false;
38480     },
38481
38482     // private
38483     deactivate : function(){
38484         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38485         this.hideMenu();
38486     },
38487
38488     // private
38489     expandMenu : function(autoActivate){
38490         if(!this.disabled && this.menu){
38491             clearTimeout(this.hideTimer);
38492             delete this.hideTimer;
38493             if(!this.menu.isVisible() && !this.showTimer){
38494                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38495             }else if (this.menu.isVisible() && autoActivate){
38496                 this.menu.tryActivate(0, 1);
38497             }
38498         }
38499     },
38500
38501     // private
38502     deferExpand : function(autoActivate){
38503         delete this.showTimer;
38504         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38505         if(autoActivate){
38506             this.menu.tryActivate(0, 1);
38507         }
38508     },
38509
38510     // private
38511     hideMenu : function(){
38512         clearTimeout(this.showTimer);
38513         delete this.showTimer;
38514         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38515             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38516         }
38517     },
38518
38519     // private
38520     deferHide : function(){
38521         delete this.hideTimer;
38522         this.menu.hide();
38523     }
38524 });/*
38525  * Based on:
38526  * Ext JS Library 1.1.1
38527  * Copyright(c) 2006-2007, Ext JS, LLC.
38528  *
38529  * Originally Released Under LGPL - original licence link has changed is not relivant.
38530  *
38531  * Fork - LGPL
38532  * <script type="text/javascript">
38533  */
38534  
38535 /**
38536  * @class Roo.menu.CheckItem
38537  * @extends Roo.menu.Item
38538  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38539  * @constructor
38540  * Creates a new CheckItem
38541  * @param {Object} config Configuration options
38542  */
38543 Roo.menu.CheckItem = function(config){
38544     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38545     this.addEvents({
38546         /**
38547          * @event beforecheckchange
38548          * Fires before the checked value is set, providing an opportunity to cancel if needed
38549          * @param {Roo.menu.CheckItem} this
38550          * @param {Boolean} checked The new checked value that will be set
38551          */
38552         "beforecheckchange" : true,
38553         /**
38554          * @event checkchange
38555          * Fires after the checked value has been set
38556          * @param {Roo.menu.CheckItem} this
38557          * @param {Boolean} checked The checked value that was set
38558          */
38559         "checkchange" : true
38560     });
38561     if(this.checkHandler){
38562         this.on('checkchange', this.checkHandler, this.scope);
38563     }
38564 };
38565 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38566     /**
38567      * @cfg {String} group
38568      * All check items with the same group name will automatically be grouped into a single-select
38569      * radio button group (defaults to '')
38570      */
38571     /**
38572      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38573      */
38574     itemCls : "x-menu-item x-menu-check-item",
38575     /**
38576      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38577      */
38578     groupClass : "x-menu-group-item",
38579
38580     /**
38581      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38582      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38583      * initialized with checked = true will be rendered as checked.
38584      */
38585     checked: false,
38586
38587     // private
38588     ctype: "Roo.menu.CheckItem",
38589
38590     // private
38591     onRender : function(c){
38592         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38593         if(this.group){
38594             this.el.addClass(this.groupClass);
38595         }
38596         Roo.menu.MenuMgr.registerCheckable(this);
38597         if(this.checked){
38598             this.checked = false;
38599             this.setChecked(true, true);
38600         }
38601     },
38602
38603     // private
38604     destroy : function(){
38605         if(this.rendered){
38606             Roo.menu.MenuMgr.unregisterCheckable(this);
38607         }
38608         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38609     },
38610
38611     /**
38612      * Set the checked state of this item
38613      * @param {Boolean} checked The new checked value
38614      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38615      */
38616     setChecked : function(state, suppressEvent){
38617         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38618             if(this.container){
38619                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38620             }
38621             this.checked = state;
38622             if(suppressEvent !== true){
38623                 this.fireEvent("checkchange", this, state);
38624             }
38625         }
38626     },
38627
38628     // private
38629     handleClick : function(e){
38630        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38631            this.setChecked(!this.checked);
38632        }
38633        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38634     }
38635 });/*
38636  * Based on:
38637  * Ext JS Library 1.1.1
38638  * Copyright(c) 2006-2007, Ext JS, LLC.
38639  *
38640  * Originally Released Under LGPL - original licence link has changed is not relivant.
38641  *
38642  * Fork - LGPL
38643  * <script type="text/javascript">
38644  */
38645  
38646 /**
38647  * @class Roo.menu.DateItem
38648  * @extends Roo.menu.Adapter
38649  * A menu item that wraps the {@link Roo.DatPicker} component.
38650  * @constructor
38651  * Creates a new DateItem
38652  * @param {Object} config Configuration options
38653  */
38654 Roo.menu.DateItem = function(config){
38655     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38656     /** The Roo.DatePicker object @type Roo.DatePicker */
38657     this.picker = this.component;
38658     this.addEvents({select: true});
38659     
38660     this.picker.on("render", function(picker){
38661         picker.getEl().swallowEvent("click");
38662         picker.container.addClass("x-menu-date-item");
38663     });
38664
38665     this.picker.on("select", this.onSelect, this);
38666 };
38667
38668 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38669     // private
38670     onSelect : function(picker, date){
38671         this.fireEvent("select", this, date, picker);
38672         Roo.menu.DateItem.superclass.handleClick.call(this);
38673     }
38674 });/*
38675  * Based on:
38676  * Ext JS Library 1.1.1
38677  * Copyright(c) 2006-2007, Ext JS, LLC.
38678  *
38679  * Originally Released Under LGPL - original licence link has changed is not relivant.
38680  *
38681  * Fork - LGPL
38682  * <script type="text/javascript">
38683  */
38684  
38685 /**
38686  * @class Roo.menu.ColorItem
38687  * @extends Roo.menu.Adapter
38688  * A menu item that wraps the {@link Roo.ColorPalette} component.
38689  * @constructor
38690  * Creates a new ColorItem
38691  * @param {Object} config Configuration options
38692  */
38693 Roo.menu.ColorItem = function(config){
38694     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38695     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38696     this.palette = this.component;
38697     this.relayEvents(this.palette, ["select"]);
38698     if(this.selectHandler){
38699         this.on('select', this.selectHandler, this.scope);
38700     }
38701 };
38702 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38703  * Based on:
38704  * Ext JS Library 1.1.1
38705  * Copyright(c) 2006-2007, Ext JS, LLC.
38706  *
38707  * Originally Released Under LGPL - original licence link has changed is not relivant.
38708  *
38709  * Fork - LGPL
38710  * <script type="text/javascript">
38711  */
38712  
38713
38714 /**
38715  * @class Roo.menu.DateMenu
38716  * @extends Roo.menu.Menu
38717  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38718  * @constructor
38719  * Creates a new DateMenu
38720  * @param {Object} config Configuration options
38721  */
38722 Roo.menu.DateMenu = function(config){
38723     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38724     this.plain = true;
38725     var di = new Roo.menu.DateItem(config);
38726     this.add(di);
38727     /**
38728      * The {@link Roo.DatePicker} instance for this DateMenu
38729      * @type DatePicker
38730      */
38731     this.picker = di.picker;
38732     /**
38733      * @event select
38734      * @param {DatePicker} picker
38735      * @param {Date} date
38736      */
38737     this.relayEvents(di, ["select"]);
38738     this.on('beforeshow', function(){
38739         if(this.picker){
38740             this.picker.hideMonthPicker(false);
38741         }
38742     }, this);
38743 };
38744 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38745     cls:'x-date-menu'
38746 });/*
38747  * Based on:
38748  * Ext JS Library 1.1.1
38749  * Copyright(c) 2006-2007, Ext JS, LLC.
38750  *
38751  * Originally Released Under LGPL - original licence link has changed is not relivant.
38752  *
38753  * Fork - LGPL
38754  * <script type="text/javascript">
38755  */
38756  
38757
38758 /**
38759  * @class Roo.menu.ColorMenu
38760  * @extends Roo.menu.Menu
38761  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38762  * @constructor
38763  * Creates a new ColorMenu
38764  * @param {Object} config Configuration options
38765  */
38766 Roo.menu.ColorMenu = function(config){
38767     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38768     this.plain = true;
38769     var ci = new Roo.menu.ColorItem(config);
38770     this.add(ci);
38771     /**
38772      * The {@link Roo.ColorPalette} instance for this ColorMenu
38773      * @type ColorPalette
38774      */
38775     this.palette = ci.palette;
38776     /**
38777      * @event select
38778      * @param {ColorPalette} palette
38779      * @param {String} color
38780      */
38781     this.relayEvents(ci, ["select"]);
38782 };
38783 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38784  * Based on:
38785  * Ext JS Library 1.1.1
38786  * Copyright(c) 2006-2007, Ext JS, LLC.
38787  *
38788  * Originally Released Under LGPL - original licence link has changed is not relivant.
38789  *
38790  * Fork - LGPL
38791  * <script type="text/javascript">
38792  */
38793  
38794 /**
38795  * @class Roo.form.TextItem
38796  * @extends Roo.BoxComponent
38797  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38798  * @constructor
38799  * Creates a new TextItem
38800  * @param {Object} config Configuration options
38801  */
38802 Roo.form.TextItem = function(config){
38803     Roo.form.TextItem.superclass.constructor.call(this, config);
38804 };
38805
38806 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38807     
38808     /**
38809      * @cfg {String} tag the tag for this item (default div)
38810      */
38811     tag : 'div',
38812     /**
38813      * @cfg {String} html the content for this item
38814      */
38815     html : '',
38816     
38817     getAutoCreate : function()
38818     {
38819         var cfg = {
38820             id: this.id,
38821             tag: this.tag,
38822             html: this.html,
38823             cls: 'x-form-item'
38824         };
38825         
38826         return cfg;
38827         
38828     },
38829     
38830     onRender : function(ct, position)
38831     {
38832         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38833         
38834         if(!this.el){
38835             var cfg = this.getAutoCreate();
38836             if(!cfg.name){
38837                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38838             }
38839             if (!cfg.name.length) {
38840                 delete cfg.name;
38841             }
38842             this.el = ct.createChild(cfg, position);
38843         }
38844     }
38845     
38846 });/*
38847  * Based on:
38848  * Ext JS Library 1.1.1
38849  * Copyright(c) 2006-2007, Ext JS, LLC.
38850  *
38851  * Originally Released Under LGPL - original licence link has changed is not relivant.
38852  *
38853  * Fork - LGPL
38854  * <script type="text/javascript">
38855  */
38856  
38857 /**
38858  * @class Roo.form.Field
38859  * @extends Roo.BoxComponent
38860  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38861  * @constructor
38862  * Creates a new Field
38863  * @param {Object} config Configuration options
38864  */
38865 Roo.form.Field = function(config){
38866     Roo.form.Field.superclass.constructor.call(this, config);
38867 };
38868
38869 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38870     /**
38871      * @cfg {String} fieldLabel Label to use when rendering a form.
38872      */
38873        /**
38874      * @cfg {String} qtip Mouse over tip
38875      */
38876      
38877     /**
38878      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38879      */
38880     invalidClass : "x-form-invalid",
38881     /**
38882      * @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")
38883      */
38884     invalidText : "The value in this field is invalid",
38885     /**
38886      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38887      */
38888     focusClass : "x-form-focus",
38889     /**
38890      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38891       automatic validation (defaults to "keyup").
38892      */
38893     validationEvent : "keyup",
38894     /**
38895      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38896      */
38897     validateOnBlur : true,
38898     /**
38899      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38900      */
38901     validationDelay : 250,
38902     /**
38903      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38904      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38905      */
38906     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38907     /**
38908      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38909      */
38910     fieldClass : "x-form-field",
38911     /**
38912      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38913      *<pre>
38914 Value         Description
38915 -----------   ----------------------------------------------------------------------
38916 qtip          Display a quick tip when the user hovers over the field
38917 title         Display a default browser title attribute popup
38918 under         Add a block div beneath the field containing the error text
38919 side          Add an error icon to the right of the field with a popup on hover
38920 [element id]  Add the error text directly to the innerHTML of the specified element
38921 </pre>
38922      */
38923     msgTarget : 'qtip',
38924     /**
38925      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38926      */
38927     msgFx : 'normal',
38928
38929     /**
38930      * @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.
38931      */
38932     readOnly : false,
38933
38934     /**
38935      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38936      */
38937     disabled : false,
38938
38939     /**
38940      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38941      */
38942     inputType : undefined,
38943     
38944     /**
38945      * @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).
38946          */
38947         tabIndex : undefined,
38948         
38949     // private
38950     isFormField : true,
38951
38952     // private
38953     hasFocus : false,
38954     /**
38955      * @property {Roo.Element} fieldEl
38956      * Element Containing the rendered Field (with label etc.)
38957      */
38958     /**
38959      * @cfg {Mixed} value A value to initialize this field with.
38960      */
38961     value : undefined,
38962
38963     /**
38964      * @cfg {String} name The field's HTML name attribute.
38965      */
38966     /**
38967      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38968      */
38969     // private
38970     loadedValue : false,
38971      
38972      
38973         // private ??
38974         initComponent : function(){
38975         Roo.form.Field.superclass.initComponent.call(this);
38976         this.addEvents({
38977             /**
38978              * @event focus
38979              * Fires when this field receives input focus.
38980              * @param {Roo.form.Field} this
38981              */
38982             focus : true,
38983             /**
38984              * @event blur
38985              * Fires when this field loses input focus.
38986              * @param {Roo.form.Field} this
38987              */
38988             blur : true,
38989             /**
38990              * @event specialkey
38991              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38992              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38993              * @param {Roo.form.Field} this
38994              * @param {Roo.EventObject} e The event object
38995              */
38996             specialkey : true,
38997             /**
38998              * @event change
38999              * Fires just before the field blurs if the field value has changed.
39000              * @param {Roo.form.Field} this
39001              * @param {Mixed} newValue The new value
39002              * @param {Mixed} oldValue The original value
39003              */
39004             change : true,
39005             /**
39006              * @event invalid
39007              * Fires after the field has been marked as invalid.
39008              * @param {Roo.form.Field} this
39009              * @param {String} msg The validation message
39010              */
39011             invalid : true,
39012             /**
39013              * @event valid
39014              * Fires after the field has been validated with no errors.
39015              * @param {Roo.form.Field} this
39016              */
39017             valid : true,
39018              /**
39019              * @event keyup
39020              * Fires after the key up
39021              * @param {Roo.form.Field} this
39022              * @param {Roo.EventObject}  e The event Object
39023              */
39024             keyup : true
39025         });
39026     },
39027
39028     /**
39029      * Returns the name attribute of the field if available
39030      * @return {String} name The field name
39031      */
39032     getName: function(){
39033          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
39034     },
39035
39036     // private
39037     onRender : function(ct, position){
39038         Roo.form.Field.superclass.onRender.call(this, ct, position);
39039         if(!this.el){
39040             var cfg = this.getAutoCreate();
39041             if(!cfg.name){
39042                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39043             }
39044             if (!cfg.name.length) {
39045                 delete cfg.name;
39046             }
39047             if(this.inputType){
39048                 cfg.type = this.inputType;
39049             }
39050             this.el = ct.createChild(cfg, position);
39051         }
39052         var type = this.el.dom.type;
39053         if(type){
39054             if(type == 'password'){
39055                 type = 'text';
39056             }
39057             this.el.addClass('x-form-'+type);
39058         }
39059         if(this.readOnly){
39060             this.el.dom.readOnly = true;
39061         }
39062         if(this.tabIndex !== undefined){
39063             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39064         }
39065
39066         this.el.addClass([this.fieldClass, this.cls]);
39067         this.initValue();
39068     },
39069
39070     /**
39071      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39072      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39073      * @return {Roo.form.Field} this
39074      */
39075     applyTo : function(target){
39076         this.allowDomMove = false;
39077         this.el = Roo.get(target);
39078         this.render(this.el.dom.parentNode);
39079         return this;
39080     },
39081
39082     // private
39083     initValue : function(){
39084         if(this.value !== undefined){
39085             this.setValue(this.value);
39086         }else if(this.el.dom.value.length > 0){
39087             this.setValue(this.el.dom.value);
39088         }
39089     },
39090
39091     /**
39092      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39093      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39094      */
39095     isDirty : function() {
39096         if(this.disabled) {
39097             return false;
39098         }
39099         return String(this.getValue()) !== String(this.originalValue);
39100     },
39101
39102     /**
39103      * stores the current value in loadedValue
39104      */
39105     resetHasChanged : function()
39106     {
39107         this.loadedValue = String(this.getValue());
39108     },
39109     /**
39110      * checks the current value against the 'loaded' value.
39111      * Note - will return false if 'resetHasChanged' has not been called first.
39112      */
39113     hasChanged : function()
39114     {
39115         if(this.disabled || this.readOnly) {
39116             return false;
39117         }
39118         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39119     },
39120     
39121     
39122     
39123     // private
39124     afterRender : function(){
39125         Roo.form.Field.superclass.afterRender.call(this);
39126         this.initEvents();
39127     },
39128
39129     // private
39130     fireKey : function(e){
39131         //Roo.log('field ' + e.getKey());
39132         if(e.isNavKeyPress()){
39133             this.fireEvent("specialkey", this, e);
39134         }
39135     },
39136
39137     /**
39138      * Resets the current field value to the originally loaded value and clears any validation messages
39139      */
39140     reset : function(){
39141         this.setValue(this.resetValue);
39142         this.originalValue = this.getValue();
39143         this.clearInvalid();
39144     },
39145
39146     // private
39147     initEvents : function(){
39148         // safari killled keypress - so keydown is now used..
39149         this.el.on("keydown" , this.fireKey,  this);
39150         this.el.on("focus", this.onFocus,  this);
39151         this.el.on("blur", this.onBlur,  this);
39152         this.el.relayEvent('keyup', this);
39153
39154         // reference to original value for reset
39155         this.originalValue = this.getValue();
39156         this.resetValue =  this.getValue();
39157     },
39158
39159     // private
39160     onFocus : function(){
39161         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39162             this.el.addClass(this.focusClass);
39163         }
39164         if(!this.hasFocus){
39165             this.hasFocus = true;
39166             this.startValue = this.getValue();
39167             this.fireEvent("focus", this);
39168         }
39169     },
39170
39171     beforeBlur : Roo.emptyFn,
39172
39173     // private
39174     onBlur : function(){
39175         this.beforeBlur();
39176         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39177             this.el.removeClass(this.focusClass);
39178         }
39179         this.hasFocus = false;
39180         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39181             this.validate();
39182         }
39183         var v = this.getValue();
39184         if(String(v) !== String(this.startValue)){
39185             this.fireEvent('change', this, v, this.startValue);
39186         }
39187         this.fireEvent("blur", this);
39188     },
39189
39190     /**
39191      * Returns whether or not the field value is currently valid
39192      * @param {Boolean} preventMark True to disable marking the field invalid
39193      * @return {Boolean} True if the value is valid, else false
39194      */
39195     isValid : function(preventMark){
39196         if(this.disabled){
39197             return true;
39198         }
39199         var restore = this.preventMark;
39200         this.preventMark = preventMark === true;
39201         var v = this.validateValue(this.processValue(this.getRawValue()));
39202         this.preventMark = restore;
39203         return v;
39204     },
39205
39206     /**
39207      * Validates the field value
39208      * @return {Boolean} True if the value is valid, else false
39209      */
39210     validate : function(){
39211         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39212             this.clearInvalid();
39213             return true;
39214         }
39215         return false;
39216     },
39217
39218     processValue : function(value){
39219         return value;
39220     },
39221
39222     // private
39223     // Subclasses should provide the validation implementation by overriding this
39224     validateValue : function(value){
39225         return true;
39226     },
39227
39228     /**
39229      * Mark this field as invalid
39230      * @param {String} msg The validation message
39231      */
39232     markInvalid : function(msg){
39233         if(!this.rendered || this.preventMark){ // not rendered
39234             return;
39235         }
39236         
39237         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39238         
39239         obj.el.addClass(this.invalidClass);
39240         msg = msg || this.invalidText;
39241         switch(this.msgTarget){
39242             case 'qtip':
39243                 obj.el.dom.qtip = msg;
39244                 obj.el.dom.qclass = 'x-form-invalid-tip';
39245                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39246                     Roo.QuickTips.enable();
39247                 }
39248                 break;
39249             case 'title':
39250                 this.el.dom.title = msg;
39251                 break;
39252             case 'under':
39253                 if(!this.errorEl){
39254                     var elp = this.el.findParent('.x-form-element', 5, true);
39255                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39256                     this.errorEl.setWidth(elp.getWidth(true)-20);
39257                 }
39258                 this.errorEl.update(msg);
39259                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39260                 break;
39261             case 'side':
39262                 if(!this.errorIcon){
39263                     var elp = this.el.findParent('.x-form-element', 5, true);
39264                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39265                 }
39266                 this.alignErrorIcon();
39267                 this.errorIcon.dom.qtip = msg;
39268                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39269                 this.errorIcon.show();
39270                 this.on('resize', this.alignErrorIcon, this);
39271                 break;
39272             default:
39273                 var t = Roo.getDom(this.msgTarget);
39274                 t.innerHTML = msg;
39275                 t.style.display = this.msgDisplay;
39276                 break;
39277         }
39278         this.fireEvent('invalid', this, msg);
39279     },
39280
39281     // private
39282     alignErrorIcon : function(){
39283         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39284     },
39285
39286     /**
39287      * Clear any invalid styles/messages for this field
39288      */
39289     clearInvalid : function(){
39290         if(!this.rendered || this.preventMark){ // not rendered
39291             return;
39292         }
39293         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39294         
39295         obj.el.removeClass(this.invalidClass);
39296         switch(this.msgTarget){
39297             case 'qtip':
39298                 obj.el.dom.qtip = '';
39299                 break;
39300             case 'title':
39301                 this.el.dom.title = '';
39302                 break;
39303             case 'under':
39304                 if(this.errorEl){
39305                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39306                 }
39307                 break;
39308             case 'side':
39309                 if(this.errorIcon){
39310                     this.errorIcon.dom.qtip = '';
39311                     this.errorIcon.hide();
39312                     this.un('resize', this.alignErrorIcon, this);
39313                 }
39314                 break;
39315             default:
39316                 var t = Roo.getDom(this.msgTarget);
39317                 t.innerHTML = '';
39318                 t.style.display = 'none';
39319                 break;
39320         }
39321         this.fireEvent('valid', this);
39322     },
39323
39324     /**
39325      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39326      * @return {Mixed} value The field value
39327      */
39328     getRawValue : function(){
39329         var v = this.el.getValue();
39330         
39331         return v;
39332     },
39333
39334     /**
39335      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39336      * @return {Mixed} value The field value
39337      */
39338     getValue : function(){
39339         var v = this.el.getValue();
39340          
39341         return v;
39342     },
39343
39344     /**
39345      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39346      * @param {Mixed} value The value to set
39347      */
39348     setRawValue : function(v){
39349         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39350     },
39351
39352     /**
39353      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39354      * @param {Mixed} value The value to set
39355      */
39356     setValue : function(v){
39357         this.value = v;
39358         if(this.rendered){
39359             this.el.dom.value = (v === null || v === undefined ? '' : v);
39360              this.validate();
39361         }
39362     },
39363
39364     adjustSize : function(w, h){
39365         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39366         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39367         return s;
39368     },
39369
39370     adjustWidth : function(tag, w){
39371         tag = tag.toLowerCase();
39372         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39373             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39374                 if(tag == 'input'){
39375                     return w + 2;
39376                 }
39377                 if(tag == 'textarea'){
39378                     return w-2;
39379                 }
39380             }else if(Roo.isOpera){
39381                 if(tag == 'input'){
39382                     return w + 2;
39383                 }
39384                 if(tag == 'textarea'){
39385                     return w-2;
39386                 }
39387             }
39388         }
39389         return w;
39390     }
39391 });
39392
39393
39394 // anything other than normal should be considered experimental
39395 Roo.form.Field.msgFx = {
39396     normal : {
39397         show: function(msgEl, f){
39398             msgEl.setDisplayed('block');
39399         },
39400
39401         hide : function(msgEl, f){
39402             msgEl.setDisplayed(false).update('');
39403         }
39404     },
39405
39406     slide : {
39407         show: function(msgEl, f){
39408             msgEl.slideIn('t', {stopFx:true});
39409         },
39410
39411         hide : function(msgEl, f){
39412             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39413         }
39414     },
39415
39416     slideRight : {
39417         show: function(msgEl, f){
39418             msgEl.fixDisplay();
39419             msgEl.alignTo(f.el, 'tl-tr');
39420             msgEl.slideIn('l', {stopFx:true});
39421         },
39422
39423         hide : function(msgEl, f){
39424             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39425         }
39426     }
39427 };/*
39428  * Based on:
39429  * Ext JS Library 1.1.1
39430  * Copyright(c) 2006-2007, Ext JS, LLC.
39431  *
39432  * Originally Released Under LGPL - original licence link has changed is not relivant.
39433  *
39434  * Fork - LGPL
39435  * <script type="text/javascript">
39436  */
39437  
39438
39439 /**
39440  * @class Roo.form.TextField
39441  * @extends Roo.form.Field
39442  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39443  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39444  * @constructor
39445  * Creates a new TextField
39446  * @param {Object} config Configuration options
39447  */
39448 Roo.form.TextField = function(config){
39449     Roo.form.TextField.superclass.constructor.call(this, config);
39450     this.addEvents({
39451         /**
39452          * @event autosize
39453          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39454          * according to the default logic, but this event provides a hook for the developer to apply additional
39455          * logic at runtime to resize the field if needed.
39456              * @param {Roo.form.Field} this This text field
39457              * @param {Number} width The new field width
39458              */
39459         autosize : true
39460     });
39461 };
39462
39463 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39464     /**
39465      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39466      */
39467     grow : false,
39468     /**
39469      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39470      */
39471     growMin : 30,
39472     /**
39473      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39474      */
39475     growMax : 800,
39476     /**
39477      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39478      */
39479     vtype : null,
39480     /**
39481      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39482      */
39483     maskRe : null,
39484     /**
39485      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39486      */
39487     disableKeyFilter : false,
39488     /**
39489      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39490      */
39491     allowBlank : true,
39492     /**
39493      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39494      */
39495     minLength : 0,
39496     /**
39497      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39498      */
39499     maxLength : Number.MAX_VALUE,
39500     /**
39501      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39502      */
39503     minLengthText : "The minimum length for this field is {0}",
39504     /**
39505      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39506      */
39507     maxLengthText : "The maximum length for this field is {0}",
39508     /**
39509      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39510      */
39511     selectOnFocus : false,
39512     /**
39513      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39514      */    
39515     allowLeadingSpace : false,
39516     /**
39517      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39518      */
39519     blankText : "This field is required",
39520     /**
39521      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39522      * If available, this function will be called only after the basic validators all return true, and will be passed the
39523      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39524      */
39525     validator : null,
39526     /**
39527      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39528      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39529      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39530      */
39531     regex : null,
39532     /**
39533      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39534      */
39535     regexText : "",
39536     /**
39537      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39538      */
39539     emptyText : null,
39540    
39541
39542     // private
39543     initEvents : function()
39544     {
39545         if (this.emptyText) {
39546             this.el.attr('placeholder', this.emptyText);
39547         }
39548         
39549         Roo.form.TextField.superclass.initEvents.call(this);
39550         if(this.validationEvent == 'keyup'){
39551             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39552             this.el.on('keyup', this.filterValidation, this);
39553         }
39554         else if(this.validationEvent !== false){
39555             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39556         }
39557         
39558         if(this.selectOnFocus){
39559             this.on("focus", this.preFocus, this);
39560         }
39561         if (!this.allowLeadingSpace) {
39562             this.on('blur', this.cleanLeadingSpace, this);
39563         }
39564         
39565         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39566             this.el.on("keypress", this.filterKeys, this);
39567         }
39568         if(this.grow){
39569             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39570             this.el.on("click", this.autoSize,  this);
39571         }
39572         if(this.el.is('input[type=password]') && Roo.isSafari){
39573             this.el.on('keydown', this.SafariOnKeyDown, this);
39574         }
39575     },
39576
39577     processValue : function(value){
39578         if(this.stripCharsRe){
39579             var newValue = value.replace(this.stripCharsRe, '');
39580             if(newValue !== value){
39581                 this.setRawValue(newValue);
39582                 return newValue;
39583             }
39584         }
39585         return value;
39586     },
39587
39588     filterValidation : function(e){
39589         if(!e.isNavKeyPress()){
39590             this.validationTask.delay(this.validationDelay);
39591         }
39592     },
39593
39594     // private
39595     onKeyUp : function(e){
39596         if(!e.isNavKeyPress()){
39597             this.autoSize();
39598         }
39599     },
39600     // private - clean the leading white space
39601     cleanLeadingSpace : function(e)
39602     {
39603         if ( this.inputType == 'file') {
39604             return;
39605         }
39606         
39607         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39608     },
39609     /**
39610      * Resets the current field value to the originally-loaded value and clears any validation messages.
39611      *  
39612      */
39613     reset : function(){
39614         Roo.form.TextField.superclass.reset.call(this);
39615        
39616     }, 
39617     // private
39618     preFocus : function(){
39619         
39620         if(this.selectOnFocus){
39621             this.el.dom.select();
39622         }
39623     },
39624
39625     
39626     // private
39627     filterKeys : function(e){
39628         var k = e.getKey();
39629         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39630             return;
39631         }
39632         var c = e.getCharCode(), cc = String.fromCharCode(c);
39633         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39634             return;
39635         }
39636         if(!this.maskRe.test(cc)){
39637             e.stopEvent();
39638         }
39639     },
39640
39641     setValue : function(v){
39642         
39643         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39644         
39645         this.autoSize();
39646     },
39647
39648     /**
39649      * Validates a value according to the field's validation rules and marks the field as invalid
39650      * if the validation fails
39651      * @param {Mixed} value The value to validate
39652      * @return {Boolean} True if the value is valid, else false
39653      */
39654     validateValue : function(value){
39655         if(value.length < 1)  { // if it's blank
39656              if(this.allowBlank){
39657                 this.clearInvalid();
39658                 return true;
39659              }else{
39660                 this.markInvalid(this.blankText);
39661                 return false;
39662              }
39663         }
39664         if(value.length < this.minLength){
39665             this.markInvalid(String.format(this.minLengthText, this.minLength));
39666             return false;
39667         }
39668         if(value.length > this.maxLength){
39669             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39670             return false;
39671         }
39672         if(this.vtype){
39673             var vt = Roo.form.VTypes;
39674             if(!vt[this.vtype](value, this)){
39675                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39676                 return false;
39677             }
39678         }
39679         if(typeof this.validator == "function"){
39680             var msg = this.validator(value);
39681             if(msg !== true){
39682                 this.markInvalid(msg);
39683                 return false;
39684             }
39685         }
39686         if(this.regex && !this.regex.test(value)){
39687             this.markInvalid(this.regexText);
39688             return false;
39689         }
39690         return true;
39691     },
39692
39693     /**
39694      * Selects text in this field
39695      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39696      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39697      */
39698     selectText : function(start, end){
39699         var v = this.getRawValue();
39700         if(v.length > 0){
39701             start = start === undefined ? 0 : start;
39702             end = end === undefined ? v.length : end;
39703             var d = this.el.dom;
39704             if(d.setSelectionRange){
39705                 d.setSelectionRange(start, end);
39706             }else if(d.createTextRange){
39707                 var range = d.createTextRange();
39708                 range.moveStart("character", start);
39709                 range.moveEnd("character", v.length-end);
39710                 range.select();
39711             }
39712         }
39713     },
39714
39715     /**
39716      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39717      * This only takes effect if grow = true, and fires the autosize event.
39718      */
39719     autoSize : function(){
39720         if(!this.grow || !this.rendered){
39721             return;
39722         }
39723         if(!this.metrics){
39724             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39725         }
39726         var el = this.el;
39727         var v = el.dom.value;
39728         var d = document.createElement('div');
39729         d.appendChild(document.createTextNode(v));
39730         v = d.innerHTML;
39731         d = null;
39732         v += "&#160;";
39733         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39734         this.el.setWidth(w);
39735         this.fireEvent("autosize", this, w);
39736     },
39737     
39738     // private
39739     SafariOnKeyDown : function(event)
39740     {
39741         // this is a workaround for a password hang bug on chrome/ webkit.
39742         
39743         var isSelectAll = false;
39744         
39745         if(this.el.dom.selectionEnd > 0){
39746             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39747         }
39748         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39749             event.preventDefault();
39750             this.setValue('');
39751             return;
39752         }
39753         
39754         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39755             
39756             event.preventDefault();
39757             // this is very hacky as keydown always get's upper case.
39758             
39759             var cc = String.fromCharCode(event.getCharCode());
39760             
39761             
39762             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39763             
39764         }
39765         
39766         
39767     }
39768 });/*
39769  * Based on:
39770  * Ext JS Library 1.1.1
39771  * Copyright(c) 2006-2007, Ext JS, LLC.
39772  *
39773  * Originally Released Under LGPL - original licence link has changed is not relivant.
39774  *
39775  * Fork - LGPL
39776  * <script type="text/javascript">
39777  */
39778  
39779 /**
39780  * @class Roo.form.Hidden
39781  * @extends Roo.form.TextField
39782  * Simple Hidden element used on forms 
39783  * 
39784  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39785  * 
39786  * @constructor
39787  * Creates a new Hidden form element.
39788  * @param {Object} config Configuration options
39789  */
39790
39791
39792
39793 // easy hidden field...
39794 Roo.form.Hidden = function(config){
39795     Roo.form.Hidden.superclass.constructor.call(this, config);
39796 };
39797   
39798 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39799     fieldLabel:      '',
39800     inputType:      'hidden',
39801     width:          50,
39802     allowBlank:     true,
39803     labelSeparator: '',
39804     hidden:         true,
39805     itemCls :       'x-form-item-display-none'
39806
39807
39808 });
39809
39810
39811 /*
39812  * Based on:
39813  * Ext JS Library 1.1.1
39814  * Copyright(c) 2006-2007, Ext JS, LLC.
39815  *
39816  * Originally Released Under LGPL - original licence link has changed is not relivant.
39817  *
39818  * Fork - LGPL
39819  * <script type="text/javascript">
39820  */
39821  
39822 /**
39823  * @class Roo.form.TriggerField
39824  * @extends Roo.form.TextField
39825  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39826  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39827  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39828  * for which you can provide a custom implementation.  For example:
39829  * <pre><code>
39830 var trigger = new Roo.form.TriggerField();
39831 trigger.onTriggerClick = myTriggerFn;
39832 trigger.applyTo('my-field');
39833 </code></pre>
39834  *
39835  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39836  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39837  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39838  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39839  * @constructor
39840  * Create a new TriggerField.
39841  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39842  * to the base TextField)
39843  */
39844 Roo.form.TriggerField = function(config){
39845     this.mimicing = false;
39846     Roo.form.TriggerField.superclass.constructor.call(this, config);
39847 };
39848
39849 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39850     /**
39851      * @cfg {String} triggerClass A CSS class to apply to the trigger
39852      */
39853     /**
39854      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39855      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39856      */
39857     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39858     /**
39859      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39860      */
39861     hideTrigger:false,
39862
39863     /** @cfg {Boolean} grow @hide */
39864     /** @cfg {Number} growMin @hide */
39865     /** @cfg {Number} growMax @hide */
39866
39867     /**
39868      * @hide 
39869      * @method
39870      */
39871     autoSize: Roo.emptyFn,
39872     // private
39873     monitorTab : true,
39874     // private
39875     deferHeight : true,
39876
39877     
39878     actionMode : 'wrap',
39879     // private
39880     onResize : function(w, h){
39881         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39882         if(typeof w == 'number'){
39883             var x = w - this.trigger.getWidth();
39884             this.el.setWidth(this.adjustWidth('input', x));
39885             this.trigger.setStyle('left', x+'px');
39886         }
39887     },
39888
39889     // private
39890     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39891
39892     // private
39893     getResizeEl : function(){
39894         return this.wrap;
39895     },
39896
39897     // private
39898     getPositionEl : function(){
39899         return this.wrap;
39900     },
39901
39902     // private
39903     alignErrorIcon : function(){
39904         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39905     },
39906
39907     // private
39908     onRender : function(ct, position){
39909         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39910         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39911         this.trigger = this.wrap.createChild(this.triggerConfig ||
39912                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39913         if(this.hideTrigger){
39914             this.trigger.setDisplayed(false);
39915         }
39916         this.initTrigger();
39917         if(!this.width){
39918             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39919         }
39920     },
39921
39922     // private
39923     initTrigger : function(){
39924         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39925         this.trigger.addClassOnOver('x-form-trigger-over');
39926         this.trigger.addClassOnClick('x-form-trigger-click');
39927     },
39928
39929     // private
39930     onDestroy : function(){
39931         if(this.trigger){
39932             this.trigger.removeAllListeners();
39933             this.trigger.remove();
39934         }
39935         if(this.wrap){
39936             this.wrap.remove();
39937         }
39938         Roo.form.TriggerField.superclass.onDestroy.call(this);
39939     },
39940
39941     // private
39942     onFocus : function(){
39943         Roo.form.TriggerField.superclass.onFocus.call(this);
39944         if(!this.mimicing){
39945             this.wrap.addClass('x-trigger-wrap-focus');
39946             this.mimicing = true;
39947             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39948             if(this.monitorTab){
39949                 this.el.on("keydown", this.checkTab, this);
39950             }
39951         }
39952     },
39953
39954     // private
39955     checkTab : function(e){
39956         if(e.getKey() == e.TAB){
39957             this.triggerBlur();
39958         }
39959     },
39960
39961     // private
39962     onBlur : function(){
39963         // do nothing
39964     },
39965
39966     // private
39967     mimicBlur : function(e, t){
39968         if(!this.wrap.contains(t) && this.validateBlur()){
39969             this.triggerBlur();
39970         }
39971     },
39972
39973     // private
39974     triggerBlur : function(){
39975         this.mimicing = false;
39976         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39977         if(this.monitorTab){
39978             this.el.un("keydown", this.checkTab, this);
39979         }
39980         this.wrap.removeClass('x-trigger-wrap-focus');
39981         Roo.form.TriggerField.superclass.onBlur.call(this);
39982     },
39983
39984     // private
39985     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39986     validateBlur : function(e, t){
39987         return true;
39988     },
39989
39990     // private
39991     onDisable : function(){
39992         Roo.form.TriggerField.superclass.onDisable.call(this);
39993         if(this.wrap){
39994             this.wrap.addClass('x-item-disabled');
39995         }
39996     },
39997
39998     // private
39999     onEnable : function(){
40000         Roo.form.TriggerField.superclass.onEnable.call(this);
40001         if(this.wrap){
40002             this.wrap.removeClass('x-item-disabled');
40003         }
40004     },
40005
40006     // private
40007     onShow : function(){
40008         var ae = this.getActionEl();
40009         
40010         if(ae){
40011             ae.dom.style.display = '';
40012             ae.dom.style.visibility = 'visible';
40013         }
40014     },
40015
40016     // private
40017     
40018     onHide : function(){
40019         var ae = this.getActionEl();
40020         ae.dom.style.display = 'none';
40021     },
40022
40023     /**
40024      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
40025      * by an implementing function.
40026      * @method
40027      * @param {EventObject} e
40028      */
40029     onTriggerClick : Roo.emptyFn
40030 });
40031
40032 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
40033 // to be extended by an implementing class.  For an example of implementing this class, see the custom
40034 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
40035 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
40036     initComponent : function(){
40037         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40038
40039         this.triggerConfig = {
40040             tag:'span', cls:'x-form-twin-triggers', cn:[
40041             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40042             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40043         ]};
40044     },
40045
40046     getTrigger : function(index){
40047         return this.triggers[index];
40048     },
40049
40050     initTrigger : function(){
40051         var ts = this.trigger.select('.x-form-trigger', true);
40052         this.wrap.setStyle('overflow', 'hidden');
40053         var triggerField = this;
40054         ts.each(function(t, all, index){
40055             t.hide = function(){
40056                 var w = triggerField.wrap.getWidth();
40057                 this.dom.style.display = 'none';
40058                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40059             };
40060             t.show = function(){
40061                 var w = triggerField.wrap.getWidth();
40062                 this.dom.style.display = '';
40063                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40064             };
40065             var triggerIndex = 'Trigger'+(index+1);
40066
40067             if(this['hide'+triggerIndex]){
40068                 t.dom.style.display = 'none';
40069             }
40070             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40071             t.addClassOnOver('x-form-trigger-over');
40072             t.addClassOnClick('x-form-trigger-click');
40073         }, this);
40074         this.triggers = ts.elements;
40075     },
40076
40077     onTrigger1Click : Roo.emptyFn,
40078     onTrigger2Click : Roo.emptyFn
40079 });/*
40080  * Based on:
40081  * Ext JS Library 1.1.1
40082  * Copyright(c) 2006-2007, Ext JS, LLC.
40083  *
40084  * Originally Released Under LGPL - original licence link has changed is not relivant.
40085  *
40086  * Fork - LGPL
40087  * <script type="text/javascript">
40088  */
40089  
40090 /**
40091  * @class Roo.form.TextArea
40092  * @extends Roo.form.TextField
40093  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40094  * support for auto-sizing.
40095  * @constructor
40096  * Creates a new TextArea
40097  * @param {Object} config Configuration options
40098  */
40099 Roo.form.TextArea = function(config){
40100     Roo.form.TextArea.superclass.constructor.call(this, config);
40101     // these are provided exchanges for backwards compat
40102     // minHeight/maxHeight were replaced by growMin/growMax to be
40103     // compatible with TextField growing config values
40104     if(this.minHeight !== undefined){
40105         this.growMin = this.minHeight;
40106     }
40107     if(this.maxHeight !== undefined){
40108         this.growMax = this.maxHeight;
40109     }
40110 };
40111
40112 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40113     /**
40114      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40115      */
40116     growMin : 60,
40117     /**
40118      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40119      */
40120     growMax: 1000,
40121     /**
40122      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40123      * in the field (equivalent to setting overflow: hidden, defaults to false)
40124      */
40125     preventScrollbars: false,
40126     /**
40127      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40128      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40129      */
40130
40131     // private
40132     onRender : function(ct, position){
40133         if(!this.el){
40134             this.defaultAutoCreate = {
40135                 tag: "textarea",
40136                 style:"width:300px;height:60px;",
40137                 autocomplete: "new-password"
40138             };
40139         }
40140         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40141         if(this.grow){
40142             this.textSizeEl = Roo.DomHelper.append(document.body, {
40143                 tag: "pre", cls: "x-form-grow-sizer"
40144             });
40145             if(this.preventScrollbars){
40146                 this.el.setStyle("overflow", "hidden");
40147             }
40148             this.el.setHeight(this.growMin);
40149         }
40150     },
40151
40152     onDestroy : function(){
40153         if(this.textSizeEl){
40154             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40155         }
40156         Roo.form.TextArea.superclass.onDestroy.call(this);
40157     },
40158
40159     // private
40160     onKeyUp : function(e){
40161         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40162             this.autoSize();
40163         }
40164     },
40165
40166     /**
40167      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40168      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40169      */
40170     autoSize : function(){
40171         if(!this.grow || !this.textSizeEl){
40172             return;
40173         }
40174         var el = this.el;
40175         var v = el.dom.value;
40176         var ts = this.textSizeEl;
40177
40178         ts.innerHTML = '';
40179         ts.appendChild(document.createTextNode(v));
40180         v = ts.innerHTML;
40181
40182         Roo.fly(ts).setWidth(this.el.getWidth());
40183         if(v.length < 1){
40184             v = "&#160;&#160;";
40185         }else{
40186             if(Roo.isIE){
40187                 v = v.replace(/\n/g, '<p>&#160;</p>');
40188             }
40189             v += "&#160;\n&#160;";
40190         }
40191         ts.innerHTML = v;
40192         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40193         if(h != this.lastHeight){
40194             this.lastHeight = h;
40195             this.el.setHeight(h);
40196             this.fireEvent("autosize", this, h);
40197         }
40198     }
40199 });/*
40200  * Based on:
40201  * Ext JS Library 1.1.1
40202  * Copyright(c) 2006-2007, Ext JS, LLC.
40203  *
40204  * Originally Released Under LGPL - original licence link has changed is not relivant.
40205  *
40206  * Fork - LGPL
40207  * <script type="text/javascript">
40208  */
40209  
40210
40211 /**
40212  * @class Roo.form.NumberField
40213  * @extends Roo.form.TextField
40214  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40215  * @constructor
40216  * Creates a new NumberField
40217  * @param {Object} config Configuration options
40218  */
40219 Roo.form.NumberField = function(config){
40220     Roo.form.NumberField.superclass.constructor.call(this, config);
40221 };
40222
40223 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40224     /**
40225      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40226      */
40227     fieldClass: "x-form-field x-form-num-field",
40228     /**
40229      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40230      */
40231     allowDecimals : true,
40232     /**
40233      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40234      */
40235     decimalSeparator : ".",
40236     /**
40237      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40238      */
40239     decimalPrecision : 2,
40240     /**
40241      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40242      */
40243     allowNegative : true,
40244     /**
40245      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40246      */
40247     minValue : Number.NEGATIVE_INFINITY,
40248     /**
40249      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40250      */
40251     maxValue : Number.MAX_VALUE,
40252     /**
40253      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40254      */
40255     minText : "The minimum value for this field is {0}",
40256     /**
40257      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40258      */
40259     maxText : "The maximum value for this field is {0}",
40260     /**
40261      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40262      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40263      */
40264     nanText : "{0} is not a valid number",
40265
40266     // private
40267     initEvents : function(){
40268         Roo.form.NumberField.superclass.initEvents.call(this);
40269         var allowed = "0123456789";
40270         if(this.allowDecimals){
40271             allowed += this.decimalSeparator;
40272         }
40273         if(this.allowNegative){
40274             allowed += "-";
40275         }
40276         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40277         var keyPress = function(e){
40278             var k = e.getKey();
40279             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40280                 return;
40281             }
40282             var c = e.getCharCode();
40283             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40284                 e.stopEvent();
40285             }
40286         };
40287         this.el.on("keypress", keyPress, this);
40288     },
40289
40290     // private
40291     validateValue : function(value){
40292         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40293             return false;
40294         }
40295         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40296              return true;
40297         }
40298         var num = this.parseValue(value);
40299         if(isNaN(num)){
40300             this.markInvalid(String.format(this.nanText, value));
40301             return false;
40302         }
40303         if(num < this.minValue){
40304             this.markInvalid(String.format(this.minText, this.minValue));
40305             return false;
40306         }
40307         if(num > this.maxValue){
40308             this.markInvalid(String.format(this.maxText, this.maxValue));
40309             return false;
40310         }
40311         return true;
40312     },
40313
40314     getValue : function(){
40315         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40316     },
40317
40318     // private
40319     parseValue : function(value){
40320         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40321         return isNaN(value) ? '' : value;
40322     },
40323
40324     // private
40325     fixPrecision : function(value){
40326         var nan = isNaN(value);
40327         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40328             return nan ? '' : value;
40329         }
40330         return parseFloat(value).toFixed(this.decimalPrecision);
40331     },
40332
40333     setValue : function(v){
40334         v = this.fixPrecision(v);
40335         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40336     },
40337
40338     // private
40339     decimalPrecisionFcn : function(v){
40340         return Math.floor(v);
40341     },
40342
40343     beforeBlur : function(){
40344         var v = this.parseValue(this.getRawValue());
40345         if(v){
40346             this.setValue(v);
40347         }
40348     }
40349 });/*
40350  * Based on:
40351  * Ext JS Library 1.1.1
40352  * Copyright(c) 2006-2007, Ext JS, LLC.
40353  *
40354  * Originally Released Under LGPL - original licence link has changed is not relivant.
40355  *
40356  * Fork - LGPL
40357  * <script type="text/javascript">
40358  */
40359  
40360 /**
40361  * @class Roo.form.DateField
40362  * @extends Roo.form.TriggerField
40363  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40364 * @constructor
40365 * Create a new DateField
40366 * @param {Object} config
40367  */
40368 Roo.form.DateField = function(config)
40369 {
40370     Roo.form.DateField.superclass.constructor.call(this, config);
40371     
40372       this.addEvents({
40373          
40374         /**
40375          * @event select
40376          * Fires when a date is selected
40377              * @param {Roo.form.DateField} combo This combo box
40378              * @param {Date} date The date selected
40379              */
40380         'select' : true
40381          
40382     });
40383     
40384     
40385     if(typeof this.minValue == "string") {
40386         this.minValue = this.parseDate(this.minValue);
40387     }
40388     if(typeof this.maxValue == "string") {
40389         this.maxValue = this.parseDate(this.maxValue);
40390     }
40391     this.ddMatch = null;
40392     if(this.disabledDates){
40393         var dd = this.disabledDates;
40394         var re = "(?:";
40395         for(var i = 0; i < dd.length; i++){
40396             re += dd[i];
40397             if(i != dd.length-1) {
40398                 re += "|";
40399             }
40400         }
40401         this.ddMatch = new RegExp(re + ")");
40402     }
40403 };
40404
40405 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40406     /**
40407      * @cfg {String} format
40408      * The default date format string which can be overriden for localization support.  The format must be
40409      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40410      */
40411     format : "m/d/y",
40412     /**
40413      * @cfg {String} altFormats
40414      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40415      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40416      */
40417     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40418     /**
40419      * @cfg {Array} disabledDays
40420      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40421      */
40422     disabledDays : null,
40423     /**
40424      * @cfg {String} disabledDaysText
40425      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40426      */
40427     disabledDaysText : "Disabled",
40428     /**
40429      * @cfg {Array} disabledDates
40430      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40431      * expression so they are very powerful. Some examples:
40432      * <ul>
40433      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40434      * <li>["03/08", "09/16"] would disable those days for every year</li>
40435      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40436      * <li>["03/../2006"] would disable every day in March 2006</li>
40437      * <li>["^03"] would disable every day in every March</li>
40438      * </ul>
40439      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40440      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40441      */
40442     disabledDates : null,
40443     /**
40444      * @cfg {String} disabledDatesText
40445      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40446      */
40447     disabledDatesText : "Disabled",
40448     /**
40449      * @cfg {Date/String} minValue
40450      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40451      * valid format (defaults to null).
40452      */
40453     minValue : null,
40454     /**
40455      * @cfg {Date/String} maxValue
40456      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40457      * valid format (defaults to null).
40458      */
40459     maxValue : null,
40460     /**
40461      * @cfg {String} minText
40462      * The error text to display when the date in the cell is before minValue (defaults to
40463      * 'The date in this field must be after {minValue}').
40464      */
40465     minText : "The date in this field must be equal to or after {0}",
40466     /**
40467      * @cfg {String} maxText
40468      * The error text to display when the date in the cell is after maxValue (defaults to
40469      * 'The date in this field must be before {maxValue}').
40470      */
40471     maxText : "The date in this field must be equal to or before {0}",
40472     /**
40473      * @cfg {String} invalidText
40474      * The error text to display when the date in the field is invalid (defaults to
40475      * '{value} is not a valid date - it must be in the format {format}').
40476      */
40477     invalidText : "{0} is not a valid date - it must be in the format {1}",
40478     /**
40479      * @cfg {String} triggerClass
40480      * An additional CSS class used to style the trigger button.  The trigger will always get the
40481      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40482      * which displays a calendar icon).
40483      */
40484     triggerClass : 'x-form-date-trigger',
40485     
40486
40487     /**
40488      * @cfg {Boolean} useIso
40489      * if enabled, then the date field will use a hidden field to store the 
40490      * real value as iso formated date. default (false)
40491      */ 
40492     useIso : false,
40493     /**
40494      * @cfg {String/Object} autoCreate
40495      * A DomHelper element spec, or true for a default element spec (defaults to
40496      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40497      */ 
40498     // private
40499     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40500     
40501     // private
40502     hiddenField: false,
40503     
40504     onRender : function(ct, position)
40505     {
40506         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40507         if (this.useIso) {
40508             //this.el.dom.removeAttribute('name'); 
40509             Roo.log("Changing name?");
40510             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40511             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40512                     'before', true);
40513             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40514             // prevent input submission
40515             this.hiddenName = this.name;
40516         }
40517             
40518             
40519     },
40520     
40521     // private
40522     validateValue : function(value)
40523     {
40524         value = this.formatDate(value);
40525         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40526             Roo.log('super failed');
40527             return false;
40528         }
40529         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40530              return true;
40531         }
40532         var svalue = value;
40533         value = this.parseDate(value);
40534         if(!value){
40535             Roo.log('parse date failed' + svalue);
40536             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40537             return false;
40538         }
40539         var time = value.getTime();
40540         if(this.minValue && time < this.minValue.getTime()){
40541             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40542             return false;
40543         }
40544         if(this.maxValue && time > this.maxValue.getTime()){
40545             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40546             return false;
40547         }
40548         if(this.disabledDays){
40549             var day = value.getDay();
40550             for(var i = 0; i < this.disabledDays.length; i++) {
40551                 if(day === this.disabledDays[i]){
40552                     this.markInvalid(this.disabledDaysText);
40553                     return false;
40554                 }
40555             }
40556         }
40557         var fvalue = this.formatDate(value);
40558         if(this.ddMatch && this.ddMatch.test(fvalue)){
40559             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40560             return false;
40561         }
40562         return true;
40563     },
40564
40565     // private
40566     // Provides logic to override the default TriggerField.validateBlur which just returns true
40567     validateBlur : function(){
40568         return !this.menu || !this.menu.isVisible();
40569     },
40570     
40571     getName: function()
40572     {
40573         // returns hidden if it's set..
40574         if (!this.rendered) {return ''};
40575         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40576         
40577     },
40578
40579     /**
40580      * Returns the current date value of the date field.
40581      * @return {Date} The date value
40582      */
40583     getValue : function(){
40584         
40585         return  this.hiddenField ?
40586                 this.hiddenField.value :
40587                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40588     },
40589
40590     /**
40591      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40592      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40593      * (the default format used is "m/d/y").
40594      * <br />Usage:
40595      * <pre><code>
40596 //All of these calls set the same date value (May 4, 2006)
40597
40598 //Pass a date object:
40599 var dt = new Date('5/4/06');
40600 dateField.setValue(dt);
40601
40602 //Pass a date string (default format):
40603 dateField.setValue('5/4/06');
40604
40605 //Pass a date string (custom format):
40606 dateField.format = 'Y-m-d';
40607 dateField.setValue('2006-5-4');
40608 </code></pre>
40609      * @param {String/Date} date The date or valid date string
40610      */
40611     setValue : function(date){
40612         if (this.hiddenField) {
40613             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40614         }
40615         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40616         // make sure the value field is always stored as a date..
40617         this.value = this.parseDate(date);
40618         
40619         
40620     },
40621
40622     // private
40623     parseDate : function(value){
40624         if(!value || value instanceof Date){
40625             return value;
40626         }
40627         var v = Date.parseDate(value, this.format);
40628          if (!v && this.useIso) {
40629             v = Date.parseDate(value, 'Y-m-d');
40630         }
40631         if(!v && this.altFormats){
40632             if(!this.altFormatsArray){
40633                 this.altFormatsArray = this.altFormats.split("|");
40634             }
40635             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40636                 v = Date.parseDate(value, this.altFormatsArray[i]);
40637             }
40638         }
40639         return v;
40640     },
40641
40642     // private
40643     formatDate : function(date, fmt){
40644         return (!date || !(date instanceof Date)) ?
40645                date : date.dateFormat(fmt || this.format);
40646     },
40647
40648     // private
40649     menuListeners : {
40650         select: function(m, d){
40651             
40652             this.setValue(d);
40653             this.fireEvent('select', this, d);
40654         },
40655         show : function(){ // retain focus styling
40656             this.onFocus();
40657         },
40658         hide : function(){
40659             this.focus.defer(10, this);
40660             var ml = this.menuListeners;
40661             this.menu.un("select", ml.select,  this);
40662             this.menu.un("show", ml.show,  this);
40663             this.menu.un("hide", ml.hide,  this);
40664         }
40665     },
40666
40667     // private
40668     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40669     onTriggerClick : function(){
40670         if(this.disabled){
40671             return;
40672         }
40673         if(this.menu == null){
40674             this.menu = new Roo.menu.DateMenu();
40675         }
40676         Roo.apply(this.menu.picker,  {
40677             showClear: this.allowBlank,
40678             minDate : this.minValue,
40679             maxDate : this.maxValue,
40680             disabledDatesRE : this.ddMatch,
40681             disabledDatesText : this.disabledDatesText,
40682             disabledDays : this.disabledDays,
40683             disabledDaysText : this.disabledDaysText,
40684             format : this.useIso ? 'Y-m-d' : this.format,
40685             minText : String.format(this.minText, this.formatDate(this.minValue)),
40686             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40687         });
40688         this.menu.on(Roo.apply({}, this.menuListeners, {
40689             scope:this
40690         }));
40691         this.menu.picker.setValue(this.getValue() || new Date());
40692         this.menu.show(this.el, "tl-bl?");
40693     },
40694
40695     beforeBlur : function(){
40696         var v = this.parseDate(this.getRawValue());
40697         if(v){
40698             this.setValue(v);
40699         }
40700     },
40701
40702     /*@
40703      * overide
40704      * 
40705      */
40706     isDirty : function() {
40707         if(this.disabled) {
40708             return false;
40709         }
40710         
40711         if(typeof(this.startValue) === 'undefined'){
40712             return false;
40713         }
40714         
40715         return String(this.getValue()) !== String(this.startValue);
40716         
40717     },
40718     // @overide
40719     cleanLeadingSpace : function(e)
40720     {
40721        return;
40722     }
40723     
40724 });/*
40725  * Based on:
40726  * Ext JS Library 1.1.1
40727  * Copyright(c) 2006-2007, Ext JS, LLC.
40728  *
40729  * Originally Released Under LGPL - original licence link has changed is not relivant.
40730  *
40731  * Fork - LGPL
40732  * <script type="text/javascript">
40733  */
40734  
40735 /**
40736  * @class Roo.form.MonthField
40737  * @extends Roo.form.TriggerField
40738  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40739 * @constructor
40740 * Create a new MonthField
40741 * @param {Object} config
40742  */
40743 Roo.form.MonthField = function(config){
40744     
40745     Roo.form.MonthField.superclass.constructor.call(this, config);
40746     
40747       this.addEvents({
40748          
40749         /**
40750          * @event select
40751          * Fires when a date is selected
40752              * @param {Roo.form.MonthFieeld} combo This combo box
40753              * @param {Date} date The date selected
40754              */
40755         'select' : true
40756          
40757     });
40758     
40759     
40760     if(typeof this.minValue == "string") {
40761         this.minValue = this.parseDate(this.minValue);
40762     }
40763     if(typeof this.maxValue == "string") {
40764         this.maxValue = this.parseDate(this.maxValue);
40765     }
40766     this.ddMatch = null;
40767     if(this.disabledDates){
40768         var dd = this.disabledDates;
40769         var re = "(?:";
40770         for(var i = 0; i < dd.length; i++){
40771             re += dd[i];
40772             if(i != dd.length-1) {
40773                 re += "|";
40774             }
40775         }
40776         this.ddMatch = new RegExp(re + ")");
40777     }
40778 };
40779
40780 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40781     /**
40782      * @cfg {String} format
40783      * The default date format string which can be overriden for localization support.  The format must be
40784      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40785      */
40786     format : "M Y",
40787     /**
40788      * @cfg {String} altFormats
40789      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40790      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40791      */
40792     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40793     /**
40794      * @cfg {Array} disabledDays
40795      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40796      */
40797     disabledDays : [0,1,2,3,4,5,6],
40798     /**
40799      * @cfg {String} disabledDaysText
40800      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40801      */
40802     disabledDaysText : "Disabled",
40803     /**
40804      * @cfg {Array} disabledDates
40805      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40806      * expression so they are very powerful. Some examples:
40807      * <ul>
40808      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40809      * <li>["03/08", "09/16"] would disable those days for every year</li>
40810      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40811      * <li>["03/../2006"] would disable every day in March 2006</li>
40812      * <li>["^03"] would disable every day in every March</li>
40813      * </ul>
40814      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40815      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40816      */
40817     disabledDates : null,
40818     /**
40819      * @cfg {String} disabledDatesText
40820      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40821      */
40822     disabledDatesText : "Disabled",
40823     /**
40824      * @cfg {Date/String} minValue
40825      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40826      * valid format (defaults to null).
40827      */
40828     minValue : null,
40829     /**
40830      * @cfg {Date/String} maxValue
40831      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40832      * valid format (defaults to null).
40833      */
40834     maxValue : null,
40835     /**
40836      * @cfg {String} minText
40837      * The error text to display when the date in the cell is before minValue (defaults to
40838      * 'The date in this field must be after {minValue}').
40839      */
40840     minText : "The date in this field must be equal to or after {0}",
40841     /**
40842      * @cfg {String} maxTextf
40843      * The error text to display when the date in the cell is after maxValue (defaults to
40844      * 'The date in this field must be before {maxValue}').
40845      */
40846     maxText : "The date in this field must be equal to or before {0}",
40847     /**
40848      * @cfg {String} invalidText
40849      * The error text to display when the date in the field is invalid (defaults to
40850      * '{value} is not a valid date - it must be in the format {format}').
40851      */
40852     invalidText : "{0} is not a valid date - it must be in the format {1}",
40853     /**
40854      * @cfg {String} triggerClass
40855      * An additional CSS class used to style the trigger button.  The trigger will always get the
40856      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40857      * which displays a calendar icon).
40858      */
40859     triggerClass : 'x-form-date-trigger',
40860     
40861
40862     /**
40863      * @cfg {Boolean} useIso
40864      * if enabled, then the date field will use a hidden field to store the 
40865      * real value as iso formated date. default (true)
40866      */ 
40867     useIso : true,
40868     /**
40869      * @cfg {String/Object} autoCreate
40870      * A DomHelper element spec, or true for a default element spec (defaults to
40871      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40872      */ 
40873     // private
40874     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40875     
40876     // private
40877     hiddenField: false,
40878     
40879     hideMonthPicker : false,
40880     
40881     onRender : function(ct, position)
40882     {
40883         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40884         if (this.useIso) {
40885             this.el.dom.removeAttribute('name'); 
40886             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40887                     'before', true);
40888             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40889             // prevent input submission
40890             this.hiddenName = this.name;
40891         }
40892             
40893             
40894     },
40895     
40896     // private
40897     validateValue : function(value)
40898     {
40899         value = this.formatDate(value);
40900         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40901             return false;
40902         }
40903         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40904              return true;
40905         }
40906         var svalue = value;
40907         value = this.parseDate(value);
40908         if(!value){
40909             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40910             return false;
40911         }
40912         var time = value.getTime();
40913         if(this.minValue && time < this.minValue.getTime()){
40914             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40915             return false;
40916         }
40917         if(this.maxValue && time > this.maxValue.getTime()){
40918             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40919             return false;
40920         }
40921         /*if(this.disabledDays){
40922             var day = value.getDay();
40923             for(var i = 0; i < this.disabledDays.length; i++) {
40924                 if(day === this.disabledDays[i]){
40925                     this.markInvalid(this.disabledDaysText);
40926                     return false;
40927                 }
40928             }
40929         }
40930         */
40931         var fvalue = this.formatDate(value);
40932         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40933             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40934             return false;
40935         }
40936         */
40937         return true;
40938     },
40939
40940     // private
40941     // Provides logic to override the default TriggerField.validateBlur which just returns true
40942     validateBlur : function(){
40943         return !this.menu || !this.menu.isVisible();
40944     },
40945
40946     /**
40947      * Returns the current date value of the date field.
40948      * @return {Date} The date value
40949      */
40950     getValue : function(){
40951         
40952         
40953         
40954         return  this.hiddenField ?
40955                 this.hiddenField.value :
40956                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40957     },
40958
40959     /**
40960      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40961      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40962      * (the default format used is "m/d/y").
40963      * <br />Usage:
40964      * <pre><code>
40965 //All of these calls set the same date value (May 4, 2006)
40966
40967 //Pass a date object:
40968 var dt = new Date('5/4/06');
40969 monthField.setValue(dt);
40970
40971 //Pass a date string (default format):
40972 monthField.setValue('5/4/06');
40973
40974 //Pass a date string (custom format):
40975 monthField.format = 'Y-m-d';
40976 monthField.setValue('2006-5-4');
40977 </code></pre>
40978      * @param {String/Date} date The date or valid date string
40979      */
40980     setValue : function(date){
40981         Roo.log('month setValue' + date);
40982         // can only be first of month..
40983         
40984         var val = this.parseDate(date);
40985         
40986         if (this.hiddenField) {
40987             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40988         }
40989         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40990         this.value = this.parseDate(date);
40991     },
40992
40993     // private
40994     parseDate : function(value){
40995         if(!value || value instanceof Date){
40996             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40997             return value;
40998         }
40999         var v = Date.parseDate(value, this.format);
41000         if (!v && this.useIso) {
41001             v = Date.parseDate(value, 'Y-m-d');
41002         }
41003         if (v) {
41004             // 
41005             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
41006         }
41007         
41008         
41009         if(!v && this.altFormats){
41010             if(!this.altFormatsArray){
41011                 this.altFormatsArray = this.altFormats.split("|");
41012             }
41013             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
41014                 v = Date.parseDate(value, this.altFormatsArray[i]);
41015             }
41016         }
41017         return v;
41018     },
41019
41020     // private
41021     formatDate : function(date, fmt){
41022         return (!date || !(date instanceof Date)) ?
41023                date : date.dateFormat(fmt || this.format);
41024     },
41025
41026     // private
41027     menuListeners : {
41028         select: function(m, d){
41029             this.setValue(d);
41030             this.fireEvent('select', this, d);
41031         },
41032         show : function(){ // retain focus styling
41033             this.onFocus();
41034         },
41035         hide : function(){
41036             this.focus.defer(10, this);
41037             var ml = this.menuListeners;
41038             this.menu.un("select", ml.select,  this);
41039             this.menu.un("show", ml.show,  this);
41040             this.menu.un("hide", ml.hide,  this);
41041         }
41042     },
41043     // private
41044     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41045     onTriggerClick : function(){
41046         if(this.disabled){
41047             return;
41048         }
41049         if(this.menu == null){
41050             this.menu = new Roo.menu.DateMenu();
41051            
41052         }
41053         
41054         Roo.apply(this.menu.picker,  {
41055             
41056             showClear: this.allowBlank,
41057             minDate : this.minValue,
41058             maxDate : this.maxValue,
41059             disabledDatesRE : this.ddMatch,
41060             disabledDatesText : this.disabledDatesText,
41061             
41062             format : this.useIso ? 'Y-m-d' : this.format,
41063             minText : String.format(this.minText, this.formatDate(this.minValue)),
41064             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41065             
41066         });
41067          this.menu.on(Roo.apply({}, this.menuListeners, {
41068             scope:this
41069         }));
41070        
41071         
41072         var m = this.menu;
41073         var p = m.picker;
41074         
41075         // hide month picker get's called when we called by 'before hide';
41076         
41077         var ignorehide = true;
41078         p.hideMonthPicker  = function(disableAnim){
41079             if (ignorehide) {
41080                 return;
41081             }
41082              if(this.monthPicker){
41083                 Roo.log("hideMonthPicker called");
41084                 if(disableAnim === true){
41085                     this.monthPicker.hide();
41086                 }else{
41087                     this.monthPicker.slideOut('t', {duration:.2});
41088                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41089                     p.fireEvent("select", this, this.value);
41090                     m.hide();
41091                 }
41092             }
41093         }
41094         
41095         Roo.log('picker set value');
41096         Roo.log(this.getValue());
41097         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41098         m.show(this.el, 'tl-bl?');
41099         ignorehide  = false;
41100         // this will trigger hideMonthPicker..
41101         
41102         
41103         // hidden the day picker
41104         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41105         
41106         
41107         
41108       
41109         
41110         p.showMonthPicker.defer(100, p);
41111     
41112         
41113        
41114     },
41115
41116     beforeBlur : function(){
41117         var v = this.parseDate(this.getRawValue());
41118         if(v){
41119             this.setValue(v);
41120         }
41121     }
41122
41123     /** @cfg {Boolean} grow @hide */
41124     /** @cfg {Number} growMin @hide */
41125     /** @cfg {Number} growMax @hide */
41126     /**
41127      * @hide
41128      * @method autoSize
41129      */
41130 });/*
41131  * Based on:
41132  * Ext JS Library 1.1.1
41133  * Copyright(c) 2006-2007, Ext JS, LLC.
41134  *
41135  * Originally Released Under LGPL - original licence link has changed is not relivant.
41136  *
41137  * Fork - LGPL
41138  * <script type="text/javascript">
41139  */
41140  
41141
41142 /**
41143  * @class Roo.form.ComboBox
41144  * @extends Roo.form.TriggerField
41145  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41146  * @constructor
41147  * Create a new ComboBox.
41148  * @param {Object} config Configuration options
41149  */
41150 Roo.form.ComboBox = function(config){
41151     Roo.form.ComboBox.superclass.constructor.call(this, config);
41152     this.addEvents({
41153         /**
41154          * @event expand
41155          * Fires when the dropdown list is expanded
41156              * @param {Roo.form.ComboBox} combo This combo box
41157              */
41158         'expand' : true,
41159         /**
41160          * @event collapse
41161          * Fires when the dropdown list is collapsed
41162              * @param {Roo.form.ComboBox} combo This combo box
41163              */
41164         'collapse' : true,
41165         /**
41166          * @event beforeselect
41167          * Fires before a list item is selected. Return false to cancel the selection.
41168              * @param {Roo.form.ComboBox} combo This combo box
41169              * @param {Roo.data.Record} record The data record returned from the underlying store
41170              * @param {Number} index The index of the selected item in the dropdown list
41171              */
41172         'beforeselect' : true,
41173         /**
41174          * @event select
41175          * Fires when a list item is selected
41176              * @param {Roo.form.ComboBox} combo This combo box
41177              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41178              * @param {Number} index The index of the selected item in the dropdown list
41179              */
41180         'select' : true,
41181         /**
41182          * @event beforequery
41183          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41184          * The event object passed has these properties:
41185              * @param {Roo.form.ComboBox} combo This combo box
41186              * @param {String} query The query
41187              * @param {Boolean} forceAll true to force "all" query
41188              * @param {Boolean} cancel true to cancel the query
41189              * @param {Object} e The query event object
41190              */
41191         'beforequery': true,
41192          /**
41193          * @event add
41194          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41195              * @param {Roo.form.ComboBox} combo This combo box
41196              */
41197         'add' : true,
41198         /**
41199          * @event edit
41200          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41201              * @param {Roo.form.ComboBox} combo This combo box
41202              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41203              */
41204         'edit' : true
41205         
41206         
41207     });
41208     if(this.transform){
41209         this.allowDomMove = false;
41210         var s = Roo.getDom(this.transform);
41211         if(!this.hiddenName){
41212             this.hiddenName = s.name;
41213         }
41214         if(!this.store){
41215             this.mode = 'local';
41216             var d = [], opts = s.options;
41217             for(var i = 0, len = opts.length;i < len; i++){
41218                 var o = opts[i];
41219                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41220                 if(o.selected) {
41221                     this.value = value;
41222                 }
41223                 d.push([value, o.text]);
41224             }
41225             this.store = new Roo.data.SimpleStore({
41226                 'id': 0,
41227                 fields: ['value', 'text'],
41228                 data : d
41229             });
41230             this.valueField = 'value';
41231             this.displayField = 'text';
41232         }
41233         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41234         if(!this.lazyRender){
41235             this.target = true;
41236             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41237             s.parentNode.removeChild(s); // remove it
41238             this.render(this.el.parentNode);
41239         }else{
41240             s.parentNode.removeChild(s); // remove it
41241         }
41242
41243     }
41244     if (this.store) {
41245         this.store = Roo.factory(this.store, Roo.data);
41246     }
41247     
41248     this.selectedIndex = -1;
41249     if(this.mode == 'local'){
41250         if(config.queryDelay === undefined){
41251             this.queryDelay = 10;
41252         }
41253         if(config.minChars === undefined){
41254             this.minChars = 0;
41255         }
41256     }
41257 };
41258
41259 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41260     /**
41261      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41262      */
41263     /**
41264      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41265      * rendering into an Roo.Editor, defaults to false)
41266      */
41267     /**
41268      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41269      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41270      */
41271     /**
41272      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41273      */
41274     /**
41275      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41276      * the dropdown list (defaults to undefined, with no header element)
41277      */
41278
41279      /**
41280      * @cfg {String/Roo.Template} tpl The template to use to render the output
41281      */
41282      
41283     // private
41284     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41285     /**
41286      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41287      */
41288     listWidth: undefined,
41289     /**
41290      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41291      * mode = 'remote' or 'text' if mode = 'local')
41292      */
41293     displayField: undefined,
41294     /**
41295      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41296      * mode = 'remote' or 'value' if mode = 'local'). 
41297      * Note: use of a valueField requires the user make a selection
41298      * in order for a value to be mapped.
41299      */
41300     valueField: undefined,
41301     
41302     
41303     /**
41304      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41305      * field's data value (defaults to the underlying DOM element's name)
41306      */
41307     hiddenName: undefined,
41308     /**
41309      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41310      */
41311     listClass: '',
41312     /**
41313      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41314      */
41315     selectedClass: 'x-combo-selected',
41316     /**
41317      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41318      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41319      * which displays a downward arrow icon).
41320      */
41321     triggerClass : 'x-form-arrow-trigger',
41322     /**
41323      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41324      */
41325     shadow:'sides',
41326     /**
41327      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41328      * anchor positions (defaults to 'tl-bl')
41329      */
41330     listAlign: 'tl-bl?',
41331     /**
41332      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41333      */
41334     maxHeight: 300,
41335     /**
41336      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41337      * query specified by the allQuery config option (defaults to 'query')
41338      */
41339     triggerAction: 'query',
41340     /**
41341      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41342      * (defaults to 4, does not apply if editable = false)
41343      */
41344     minChars : 4,
41345     /**
41346      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41347      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41348      */
41349     typeAhead: false,
41350     /**
41351      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41352      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41353      */
41354     queryDelay: 500,
41355     /**
41356      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41357      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41358      */
41359     pageSize: 0,
41360     /**
41361      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41362      * when editable = true (defaults to false)
41363      */
41364     selectOnFocus:false,
41365     /**
41366      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41367      */
41368     queryParam: 'query',
41369     /**
41370      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41371      * when mode = 'remote' (defaults to 'Loading...')
41372      */
41373     loadingText: 'Loading...',
41374     /**
41375      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41376      */
41377     resizable: false,
41378     /**
41379      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41380      */
41381     handleHeight : 8,
41382     /**
41383      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41384      * traditional select (defaults to true)
41385      */
41386     editable: true,
41387     /**
41388      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41389      */
41390     allQuery: '',
41391     /**
41392      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41393      */
41394     mode: 'remote',
41395     /**
41396      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41397      * listWidth has a higher value)
41398      */
41399     minListWidth : 70,
41400     /**
41401      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41402      * allow the user to set arbitrary text into the field (defaults to false)
41403      */
41404     forceSelection:false,
41405     /**
41406      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41407      * if typeAhead = true (defaults to 250)
41408      */
41409     typeAheadDelay : 250,
41410     /**
41411      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41412      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41413      */
41414     valueNotFoundText : undefined,
41415     /**
41416      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41417      */
41418     blockFocus : false,
41419     
41420     /**
41421      * @cfg {Boolean} disableClear Disable showing of clear button.
41422      */
41423     disableClear : false,
41424     /**
41425      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41426      */
41427     alwaysQuery : false,
41428     
41429     //private
41430     addicon : false,
41431     editicon: false,
41432     
41433     // element that contains real text value.. (when hidden is used..)
41434      
41435     // private
41436     onRender : function(ct, position)
41437     {
41438         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41439         
41440         if(this.hiddenName){
41441             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41442                     'before', true);
41443             this.hiddenField.value =
41444                 this.hiddenValue !== undefined ? this.hiddenValue :
41445                 this.value !== undefined ? this.value : '';
41446
41447             // prevent input submission
41448             this.el.dom.removeAttribute('name');
41449              
41450              
41451         }
41452         
41453         if(Roo.isGecko){
41454             this.el.dom.setAttribute('autocomplete', 'off');
41455         }
41456
41457         var cls = 'x-combo-list';
41458
41459         this.list = new Roo.Layer({
41460             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41461         });
41462
41463         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41464         this.list.setWidth(lw);
41465         this.list.swallowEvent('mousewheel');
41466         this.assetHeight = 0;
41467
41468         if(this.title){
41469             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41470             this.assetHeight += this.header.getHeight();
41471         }
41472
41473         this.innerList = this.list.createChild({cls:cls+'-inner'});
41474         this.innerList.on('mouseover', this.onViewOver, this);
41475         this.innerList.on('mousemove', this.onViewMove, this);
41476         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41477         
41478         if(this.allowBlank && !this.pageSize && !this.disableClear){
41479             this.footer = this.list.createChild({cls:cls+'-ft'});
41480             this.pageTb = new Roo.Toolbar(this.footer);
41481            
41482         }
41483         if(this.pageSize){
41484             this.footer = this.list.createChild({cls:cls+'-ft'});
41485             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41486                     {pageSize: this.pageSize});
41487             
41488         }
41489         
41490         if (this.pageTb && this.allowBlank && !this.disableClear) {
41491             var _this = this;
41492             this.pageTb.add(new Roo.Toolbar.Fill(), {
41493                 cls: 'x-btn-icon x-btn-clear',
41494                 text: '&#160;',
41495                 handler: function()
41496                 {
41497                     _this.collapse();
41498                     _this.clearValue();
41499                     _this.onSelect(false, -1);
41500                 }
41501             });
41502         }
41503         if (this.footer) {
41504             this.assetHeight += this.footer.getHeight();
41505         }
41506         
41507
41508         if(!this.tpl){
41509             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41510         }
41511
41512         this.view = new Roo.View(this.innerList, this.tpl, {
41513             singleSelect:true,
41514             store: this.store,
41515             selectedClass: this.selectedClass
41516         });
41517
41518         this.view.on('click', this.onViewClick, this);
41519
41520         this.store.on('beforeload', this.onBeforeLoad, this);
41521         this.store.on('load', this.onLoad, this);
41522         this.store.on('loadexception', this.onLoadException, this);
41523
41524         if(this.resizable){
41525             this.resizer = new Roo.Resizable(this.list,  {
41526                pinned:true, handles:'se'
41527             });
41528             this.resizer.on('resize', function(r, w, h){
41529                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41530                 this.listWidth = w;
41531                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41532                 this.restrictHeight();
41533             }, this);
41534             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41535         }
41536         if(!this.editable){
41537             this.editable = true;
41538             this.setEditable(false);
41539         }  
41540         
41541         
41542         if (typeof(this.events.add.listeners) != 'undefined') {
41543             
41544             this.addicon = this.wrap.createChild(
41545                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41546        
41547             this.addicon.on('click', function(e) {
41548                 this.fireEvent('add', this);
41549             }, this);
41550         }
41551         if (typeof(this.events.edit.listeners) != 'undefined') {
41552             
41553             this.editicon = this.wrap.createChild(
41554                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41555             if (this.addicon) {
41556                 this.editicon.setStyle('margin-left', '40px');
41557             }
41558             this.editicon.on('click', function(e) {
41559                 
41560                 // we fire even  if inothing is selected..
41561                 this.fireEvent('edit', this, this.lastData );
41562                 
41563             }, this);
41564         }
41565         
41566         
41567         
41568     },
41569
41570     // private
41571     initEvents : function(){
41572         Roo.form.ComboBox.superclass.initEvents.call(this);
41573
41574         this.keyNav = new Roo.KeyNav(this.el, {
41575             "up" : function(e){
41576                 this.inKeyMode = true;
41577                 this.selectPrev();
41578             },
41579
41580             "down" : function(e){
41581                 if(!this.isExpanded()){
41582                     this.onTriggerClick();
41583                 }else{
41584                     this.inKeyMode = true;
41585                     this.selectNext();
41586                 }
41587             },
41588
41589             "enter" : function(e){
41590                 this.onViewClick();
41591                 //return true;
41592             },
41593
41594             "esc" : function(e){
41595                 this.collapse();
41596             },
41597
41598             "tab" : function(e){
41599                 this.onViewClick(false);
41600                 this.fireEvent("specialkey", this, e);
41601                 return true;
41602             },
41603
41604             scope : this,
41605
41606             doRelay : function(foo, bar, hname){
41607                 if(hname == 'down' || this.scope.isExpanded()){
41608                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41609                 }
41610                 return true;
41611             },
41612
41613             forceKeyDown: true
41614         });
41615         this.queryDelay = Math.max(this.queryDelay || 10,
41616                 this.mode == 'local' ? 10 : 250);
41617         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41618         if(this.typeAhead){
41619             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41620         }
41621         if(this.editable !== false){
41622             this.el.on("keyup", this.onKeyUp, this);
41623         }
41624         if(this.forceSelection){
41625             this.on('blur', this.doForce, this);
41626         }
41627     },
41628
41629     onDestroy : function(){
41630         if(this.view){
41631             this.view.setStore(null);
41632             this.view.el.removeAllListeners();
41633             this.view.el.remove();
41634             this.view.purgeListeners();
41635         }
41636         if(this.list){
41637             this.list.destroy();
41638         }
41639         if(this.store){
41640             this.store.un('beforeload', this.onBeforeLoad, this);
41641             this.store.un('load', this.onLoad, this);
41642             this.store.un('loadexception', this.onLoadException, this);
41643         }
41644         Roo.form.ComboBox.superclass.onDestroy.call(this);
41645     },
41646
41647     // private
41648     fireKey : function(e){
41649         if(e.isNavKeyPress() && !this.list.isVisible()){
41650             this.fireEvent("specialkey", this, e);
41651         }
41652     },
41653
41654     // private
41655     onResize: function(w, h){
41656         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41657         
41658         if(typeof w != 'number'){
41659             // we do not handle it!?!?
41660             return;
41661         }
41662         var tw = this.trigger.getWidth();
41663         tw += this.addicon ? this.addicon.getWidth() : 0;
41664         tw += this.editicon ? this.editicon.getWidth() : 0;
41665         var x = w - tw;
41666         this.el.setWidth( this.adjustWidth('input', x));
41667             
41668         this.trigger.setStyle('left', x+'px');
41669         
41670         if(this.list && this.listWidth === undefined){
41671             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41672             this.list.setWidth(lw);
41673             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41674         }
41675         
41676     
41677         
41678     },
41679
41680     /**
41681      * Allow or prevent the user from directly editing the field text.  If false is passed,
41682      * the user will only be able to select from the items defined in the dropdown list.  This method
41683      * is the runtime equivalent of setting the 'editable' config option at config time.
41684      * @param {Boolean} value True to allow the user to directly edit the field text
41685      */
41686     setEditable : function(value){
41687         if(value == this.editable){
41688             return;
41689         }
41690         this.editable = value;
41691         if(!value){
41692             this.el.dom.setAttribute('readOnly', true);
41693             this.el.on('mousedown', this.onTriggerClick,  this);
41694             this.el.addClass('x-combo-noedit');
41695         }else{
41696             this.el.dom.setAttribute('readOnly', false);
41697             this.el.un('mousedown', this.onTriggerClick,  this);
41698             this.el.removeClass('x-combo-noedit');
41699         }
41700     },
41701
41702     // private
41703     onBeforeLoad : function(){
41704         if(!this.hasFocus){
41705             return;
41706         }
41707         this.innerList.update(this.loadingText ?
41708                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41709         this.restrictHeight();
41710         this.selectedIndex = -1;
41711     },
41712
41713     // private
41714     onLoad : function(){
41715         if(!this.hasFocus){
41716             return;
41717         }
41718         if(this.store.getCount() > 0){
41719             this.expand();
41720             this.restrictHeight();
41721             if(this.lastQuery == this.allQuery){
41722                 if(this.editable){
41723                     this.el.dom.select();
41724                 }
41725                 if(!this.selectByValue(this.value, true)){
41726                     this.select(0, true);
41727                 }
41728             }else{
41729                 this.selectNext();
41730                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41731                     this.taTask.delay(this.typeAheadDelay);
41732                 }
41733             }
41734         }else{
41735             this.onEmptyResults();
41736         }
41737         //this.el.focus();
41738     },
41739     // private
41740     onLoadException : function()
41741     {
41742         this.collapse();
41743         Roo.log(this.store.reader.jsonData);
41744         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41745             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41746         }
41747         
41748         
41749     },
41750     // private
41751     onTypeAhead : function(){
41752         if(this.store.getCount() > 0){
41753             var r = this.store.getAt(0);
41754             var newValue = r.data[this.displayField];
41755             var len = newValue.length;
41756             var selStart = this.getRawValue().length;
41757             if(selStart != len){
41758                 this.setRawValue(newValue);
41759                 this.selectText(selStart, newValue.length);
41760             }
41761         }
41762     },
41763
41764     // private
41765     onSelect : function(record, index){
41766         if(this.fireEvent('beforeselect', this, record, index) !== false){
41767             this.setFromData(index > -1 ? record.data : false);
41768             this.collapse();
41769             this.fireEvent('select', this, record, index);
41770         }
41771     },
41772
41773     /**
41774      * Returns the currently selected field value or empty string if no value is set.
41775      * @return {String} value The selected value
41776      */
41777     getValue : function(){
41778         if(this.valueField){
41779             return typeof this.value != 'undefined' ? this.value : '';
41780         }
41781         return Roo.form.ComboBox.superclass.getValue.call(this);
41782     },
41783
41784     /**
41785      * Clears any text/value currently set in the field
41786      */
41787     clearValue : function(){
41788         if(this.hiddenField){
41789             this.hiddenField.value = '';
41790         }
41791         this.value = '';
41792         this.setRawValue('');
41793         this.lastSelectionText = '';
41794         
41795     },
41796
41797     /**
41798      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41799      * will be displayed in the field.  If the value does not match the data value of an existing item,
41800      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41801      * Otherwise the field will be blank (although the value will still be set).
41802      * @param {String} value The value to match
41803      */
41804     setValue : function(v){
41805         var text = v;
41806         if(this.valueField){
41807             var r = this.findRecord(this.valueField, v);
41808             if(r){
41809                 text = r.data[this.displayField];
41810             }else if(this.valueNotFoundText !== undefined){
41811                 text = this.valueNotFoundText;
41812             }
41813         }
41814         this.lastSelectionText = text;
41815         if(this.hiddenField){
41816             this.hiddenField.value = v;
41817         }
41818         Roo.form.ComboBox.superclass.setValue.call(this, text);
41819         this.value = v;
41820     },
41821     /**
41822      * @property {Object} the last set data for the element
41823      */
41824     
41825     lastData : false,
41826     /**
41827      * Sets the value of the field based on a object which is related to the record format for the store.
41828      * @param {Object} value the value to set as. or false on reset?
41829      */
41830     setFromData : function(o){
41831         var dv = ''; // display value
41832         var vv = ''; // value value..
41833         this.lastData = o;
41834         if (this.displayField) {
41835             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41836         } else {
41837             // this is an error condition!!!
41838             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41839         }
41840         
41841         if(this.valueField){
41842             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41843         }
41844         if(this.hiddenField){
41845             this.hiddenField.value = vv;
41846             
41847             this.lastSelectionText = dv;
41848             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41849             this.value = vv;
41850             return;
41851         }
41852         // no hidden field.. - we store the value in 'value', but still display
41853         // display field!!!!
41854         this.lastSelectionText = dv;
41855         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41856         this.value = vv;
41857         
41858         
41859     },
41860     // private
41861     reset : function(){
41862         // overridden so that last data is reset..
41863         this.setValue(this.resetValue);
41864         this.originalValue = this.getValue();
41865         this.clearInvalid();
41866         this.lastData = false;
41867         if (this.view) {
41868             this.view.clearSelections();
41869         }
41870     },
41871     // private
41872     findRecord : function(prop, value){
41873         var record;
41874         if(this.store.getCount() > 0){
41875             this.store.each(function(r){
41876                 if(r.data[prop] == value){
41877                     record = r;
41878                     return false;
41879                 }
41880                 return true;
41881             });
41882         }
41883         return record;
41884     },
41885     
41886     getName: function()
41887     {
41888         // returns hidden if it's set..
41889         if (!this.rendered) {return ''};
41890         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41891         
41892     },
41893     // private
41894     onViewMove : function(e, t){
41895         this.inKeyMode = false;
41896     },
41897
41898     // private
41899     onViewOver : function(e, t){
41900         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41901             return;
41902         }
41903         var item = this.view.findItemFromChild(t);
41904         if(item){
41905             var index = this.view.indexOf(item);
41906             this.select(index, false);
41907         }
41908     },
41909
41910     // private
41911     onViewClick : function(doFocus)
41912     {
41913         var index = this.view.getSelectedIndexes()[0];
41914         var r = this.store.getAt(index);
41915         if(r){
41916             this.onSelect(r, index);
41917         }
41918         if(doFocus !== false && !this.blockFocus){
41919             this.el.focus();
41920         }
41921     },
41922
41923     // private
41924     restrictHeight : function(){
41925         this.innerList.dom.style.height = '';
41926         var inner = this.innerList.dom;
41927         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41928         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41929         this.list.beginUpdate();
41930         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41931         this.list.alignTo(this.el, this.listAlign);
41932         this.list.endUpdate();
41933     },
41934
41935     // private
41936     onEmptyResults : function(){
41937         this.collapse();
41938     },
41939
41940     /**
41941      * Returns true if the dropdown list is expanded, else false.
41942      */
41943     isExpanded : function(){
41944         return this.list.isVisible();
41945     },
41946
41947     /**
41948      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41949      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41950      * @param {String} value The data value of the item to select
41951      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41952      * selected item if it is not currently in view (defaults to true)
41953      * @return {Boolean} True if the value matched an item in the list, else false
41954      */
41955     selectByValue : function(v, scrollIntoView){
41956         if(v !== undefined && v !== null){
41957             var r = this.findRecord(this.valueField || this.displayField, v);
41958             if(r){
41959                 this.select(this.store.indexOf(r), scrollIntoView);
41960                 return true;
41961             }
41962         }
41963         return false;
41964     },
41965
41966     /**
41967      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41968      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41969      * @param {Number} index The zero-based index of the list item to select
41970      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41971      * selected item if it is not currently in view (defaults to true)
41972      */
41973     select : function(index, scrollIntoView){
41974         this.selectedIndex = index;
41975         this.view.select(index);
41976         if(scrollIntoView !== false){
41977             var el = this.view.getNode(index);
41978             if(el){
41979                 this.innerList.scrollChildIntoView(el, false);
41980             }
41981         }
41982     },
41983
41984     // private
41985     selectNext : function(){
41986         var ct = this.store.getCount();
41987         if(ct > 0){
41988             if(this.selectedIndex == -1){
41989                 this.select(0);
41990             }else if(this.selectedIndex < ct-1){
41991                 this.select(this.selectedIndex+1);
41992             }
41993         }
41994     },
41995
41996     // private
41997     selectPrev : function(){
41998         var ct = this.store.getCount();
41999         if(ct > 0){
42000             if(this.selectedIndex == -1){
42001                 this.select(0);
42002             }else if(this.selectedIndex != 0){
42003                 this.select(this.selectedIndex-1);
42004             }
42005         }
42006     },
42007
42008     // private
42009     onKeyUp : function(e){
42010         if(this.editable !== false && !e.isSpecialKey()){
42011             this.lastKey = e.getKey();
42012             this.dqTask.delay(this.queryDelay);
42013         }
42014     },
42015
42016     // private
42017     validateBlur : function(){
42018         return !this.list || !this.list.isVisible();   
42019     },
42020
42021     // private
42022     initQuery : function(){
42023         this.doQuery(this.getRawValue());
42024     },
42025
42026     // private
42027     doForce : function(){
42028         if(this.el.dom.value.length > 0){
42029             this.el.dom.value =
42030                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
42031              
42032         }
42033     },
42034
42035     /**
42036      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
42037      * query allowing the query action to be canceled if needed.
42038      * @param {String} query The SQL query to execute
42039      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42040      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42041      * saved in the current store (defaults to false)
42042      */
42043     doQuery : function(q, forceAll){
42044         if(q === undefined || q === null){
42045             q = '';
42046         }
42047         var qe = {
42048             query: q,
42049             forceAll: forceAll,
42050             combo: this,
42051             cancel:false
42052         };
42053         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42054             return false;
42055         }
42056         q = qe.query;
42057         forceAll = qe.forceAll;
42058         if(forceAll === true || (q.length >= this.minChars)){
42059             if(this.lastQuery != q || this.alwaysQuery){
42060                 this.lastQuery = q;
42061                 if(this.mode == 'local'){
42062                     this.selectedIndex = -1;
42063                     if(forceAll){
42064                         this.store.clearFilter();
42065                     }else{
42066                         this.store.filter(this.displayField, q);
42067                     }
42068                     this.onLoad();
42069                 }else{
42070                     this.store.baseParams[this.queryParam] = q;
42071                     this.store.load({
42072                         params: this.getParams(q)
42073                     });
42074                     this.expand();
42075                 }
42076             }else{
42077                 this.selectedIndex = -1;
42078                 this.onLoad();   
42079             }
42080         }
42081     },
42082
42083     // private
42084     getParams : function(q){
42085         var p = {};
42086         //p[this.queryParam] = q;
42087         if(this.pageSize){
42088             p.start = 0;
42089             p.limit = this.pageSize;
42090         }
42091         return p;
42092     },
42093
42094     /**
42095      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42096      */
42097     collapse : function(){
42098         if(!this.isExpanded()){
42099             return;
42100         }
42101         this.list.hide();
42102         Roo.get(document).un('mousedown', this.collapseIf, this);
42103         Roo.get(document).un('mousewheel', this.collapseIf, this);
42104         if (!this.editable) {
42105             Roo.get(document).un('keydown', this.listKeyPress, this);
42106         }
42107         this.fireEvent('collapse', this);
42108     },
42109
42110     // private
42111     collapseIf : function(e){
42112         if(!e.within(this.wrap) && !e.within(this.list)){
42113             this.collapse();
42114         }
42115     },
42116
42117     /**
42118      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42119      */
42120     expand : function(){
42121         if(this.isExpanded() || !this.hasFocus){
42122             return;
42123         }
42124         this.list.alignTo(this.el, this.listAlign);
42125         this.list.show();
42126         Roo.get(document).on('mousedown', this.collapseIf, this);
42127         Roo.get(document).on('mousewheel', this.collapseIf, this);
42128         if (!this.editable) {
42129             Roo.get(document).on('keydown', this.listKeyPress, this);
42130         }
42131         
42132         this.fireEvent('expand', this);
42133     },
42134
42135     // private
42136     // Implements the default empty TriggerField.onTriggerClick function
42137     onTriggerClick : function(){
42138         if(this.disabled){
42139             return;
42140         }
42141         if(this.isExpanded()){
42142             this.collapse();
42143             if (!this.blockFocus) {
42144                 this.el.focus();
42145             }
42146             
42147         }else {
42148             this.hasFocus = true;
42149             if(this.triggerAction == 'all') {
42150                 this.doQuery(this.allQuery, true);
42151             } else {
42152                 this.doQuery(this.getRawValue());
42153             }
42154             if (!this.blockFocus) {
42155                 this.el.focus();
42156             }
42157         }
42158     },
42159     listKeyPress : function(e)
42160     {
42161         //Roo.log('listkeypress');
42162         // scroll to first matching element based on key pres..
42163         if (e.isSpecialKey()) {
42164             return false;
42165         }
42166         var k = String.fromCharCode(e.getKey()).toUpperCase();
42167         //Roo.log(k);
42168         var match  = false;
42169         var csel = this.view.getSelectedNodes();
42170         var cselitem = false;
42171         if (csel.length) {
42172             var ix = this.view.indexOf(csel[0]);
42173             cselitem  = this.store.getAt(ix);
42174             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42175                 cselitem = false;
42176             }
42177             
42178         }
42179         
42180         this.store.each(function(v) { 
42181             if (cselitem) {
42182                 // start at existing selection.
42183                 if (cselitem.id == v.id) {
42184                     cselitem = false;
42185                 }
42186                 return;
42187             }
42188                 
42189             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42190                 match = this.store.indexOf(v);
42191                 return false;
42192             }
42193         }, this);
42194         
42195         if (match === false) {
42196             return true; // no more action?
42197         }
42198         // scroll to?
42199         this.view.select(match);
42200         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42201         sn.scrollIntoView(sn.dom.parentNode, false);
42202     } 
42203
42204     /** 
42205     * @cfg {Boolean} grow 
42206     * @hide 
42207     */
42208     /** 
42209     * @cfg {Number} growMin 
42210     * @hide 
42211     */
42212     /** 
42213     * @cfg {Number} growMax 
42214     * @hide 
42215     */
42216     /**
42217      * @hide
42218      * @method autoSize
42219      */
42220 });/*
42221  * Copyright(c) 2010-2012, Roo J Solutions Limited
42222  *
42223  * Licence LGPL
42224  *
42225  */
42226
42227 /**
42228  * @class Roo.form.ComboBoxArray
42229  * @extends Roo.form.TextField
42230  * A facebook style adder... for lists of email / people / countries  etc...
42231  * pick multiple items from a combo box, and shows each one.
42232  *
42233  *  Fred [x]  Brian [x]  [Pick another |v]
42234  *
42235  *
42236  *  For this to work: it needs various extra information
42237  *    - normal combo problay has
42238  *      name, hiddenName
42239  *    + displayField, valueField
42240  *
42241  *    For our purpose...
42242  *
42243  *
42244  *   If we change from 'extends' to wrapping...
42245  *   
42246  *  
42247  *
42248  
42249  
42250  * @constructor
42251  * Create a new ComboBoxArray.
42252  * @param {Object} config Configuration options
42253  */
42254  
42255
42256 Roo.form.ComboBoxArray = function(config)
42257 {
42258     this.addEvents({
42259         /**
42260          * @event beforeremove
42261          * Fires before remove the value from the list
42262              * @param {Roo.form.ComboBoxArray} _self This combo box array
42263              * @param {Roo.form.ComboBoxArray.Item} item removed item
42264              */
42265         'beforeremove' : true,
42266         /**
42267          * @event remove
42268          * Fires when remove the value from the list
42269              * @param {Roo.form.ComboBoxArray} _self This combo box array
42270              * @param {Roo.form.ComboBoxArray.Item} item removed item
42271              */
42272         'remove' : true
42273         
42274         
42275     });
42276     
42277     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42278     
42279     this.items = new Roo.util.MixedCollection(false);
42280     
42281     // construct the child combo...
42282     
42283     
42284     
42285     
42286    
42287     
42288 }
42289
42290  
42291 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42292
42293     /**
42294      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42295      */
42296     
42297     lastData : false,
42298     
42299     // behavies liek a hiddne field
42300     inputType:      'hidden',
42301     /**
42302      * @cfg {Number} width The width of the box that displays the selected element
42303      */ 
42304     width:          300,
42305
42306     
42307     
42308     /**
42309      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42310      */
42311     name : false,
42312     /**
42313      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42314      */
42315     hiddenName : false,
42316     
42317     
42318     // private the array of items that are displayed..
42319     items  : false,
42320     // private - the hidden field el.
42321     hiddenEl : false,
42322     // private - the filed el..
42323     el : false,
42324     
42325     //validateValue : function() { return true; }, // all values are ok!
42326     //onAddClick: function() { },
42327     
42328     onRender : function(ct, position) 
42329     {
42330         
42331         // create the standard hidden element
42332         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42333         
42334         
42335         // give fake names to child combo;
42336         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42337         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42338         
42339         this.combo = Roo.factory(this.combo, Roo.form);
42340         this.combo.onRender(ct, position);
42341         if (typeof(this.combo.width) != 'undefined') {
42342             this.combo.onResize(this.combo.width,0);
42343         }
42344         
42345         this.combo.initEvents();
42346         
42347         // assigned so form know we need to do this..
42348         this.store          = this.combo.store;
42349         this.valueField     = this.combo.valueField;
42350         this.displayField   = this.combo.displayField ;
42351         
42352         
42353         this.combo.wrap.addClass('x-cbarray-grp');
42354         
42355         var cbwrap = this.combo.wrap.createChild(
42356             {tag: 'div', cls: 'x-cbarray-cb'},
42357             this.combo.el.dom
42358         );
42359         
42360              
42361         this.hiddenEl = this.combo.wrap.createChild({
42362             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42363         });
42364         this.el = this.combo.wrap.createChild({
42365             tag: 'input',  type:'hidden' , name: this.name, value : ''
42366         });
42367          //   this.el.dom.removeAttribute("name");
42368         
42369         
42370         this.outerWrap = this.combo.wrap;
42371         this.wrap = cbwrap;
42372         
42373         this.outerWrap.setWidth(this.width);
42374         this.outerWrap.dom.removeChild(this.el.dom);
42375         
42376         this.wrap.dom.appendChild(this.el.dom);
42377         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42378         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42379         
42380         this.combo.trigger.setStyle('position','relative');
42381         this.combo.trigger.setStyle('left', '0px');
42382         this.combo.trigger.setStyle('top', '2px');
42383         
42384         this.combo.el.setStyle('vertical-align', 'text-bottom');
42385         
42386         //this.trigger.setStyle('vertical-align', 'top');
42387         
42388         // this should use the code from combo really... on('add' ....)
42389         if (this.adder) {
42390             
42391         
42392             this.adder = this.outerWrap.createChild(
42393                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42394             var _t = this;
42395             this.adder.on('click', function(e) {
42396                 _t.fireEvent('adderclick', this, e);
42397             }, _t);
42398         }
42399         //var _t = this;
42400         //this.adder.on('click', this.onAddClick, _t);
42401         
42402         
42403         this.combo.on('select', function(cb, rec, ix) {
42404             this.addItem(rec.data);
42405             
42406             cb.setValue('');
42407             cb.el.dom.value = '';
42408             //cb.lastData = rec.data;
42409             // add to list
42410             
42411         }, this);
42412         
42413         
42414     },
42415     
42416     
42417     getName: function()
42418     {
42419         // returns hidden if it's set..
42420         if (!this.rendered) {return ''};
42421         return  this.hiddenName ? this.hiddenName : this.name;
42422         
42423     },
42424     
42425     
42426     onResize: function(w, h){
42427         
42428         return;
42429         // not sure if this is needed..
42430         //this.combo.onResize(w,h);
42431         
42432         if(typeof w != 'number'){
42433             // we do not handle it!?!?
42434             return;
42435         }
42436         var tw = this.combo.trigger.getWidth();
42437         tw += this.addicon ? this.addicon.getWidth() : 0;
42438         tw += this.editicon ? this.editicon.getWidth() : 0;
42439         var x = w - tw;
42440         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42441             
42442         this.combo.trigger.setStyle('left', '0px');
42443         
42444         if(this.list && this.listWidth === undefined){
42445             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42446             this.list.setWidth(lw);
42447             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42448         }
42449         
42450     
42451         
42452     },
42453     
42454     addItem: function(rec)
42455     {
42456         var valueField = this.combo.valueField;
42457         var displayField = this.combo.displayField;
42458         
42459         if (this.items.indexOfKey(rec[valueField]) > -1) {
42460             //console.log("GOT " + rec.data.id);
42461             return;
42462         }
42463         
42464         var x = new Roo.form.ComboBoxArray.Item({
42465             //id : rec[this.idField],
42466             data : rec,
42467             displayField : displayField ,
42468             tipField : displayField ,
42469             cb : this
42470         });
42471         // use the 
42472         this.items.add(rec[valueField],x);
42473         // add it before the element..
42474         this.updateHiddenEl();
42475         x.render(this.outerWrap, this.wrap.dom);
42476         // add the image handler..
42477     },
42478     
42479     updateHiddenEl : function()
42480     {
42481         this.validate();
42482         if (!this.hiddenEl) {
42483             return;
42484         }
42485         var ar = [];
42486         var idField = this.combo.valueField;
42487         
42488         this.items.each(function(f) {
42489             ar.push(f.data[idField]);
42490         });
42491         this.hiddenEl.dom.value = ar.join(',');
42492         this.validate();
42493     },
42494     
42495     reset : function()
42496     {
42497         this.items.clear();
42498         
42499         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42500            el.remove();
42501         });
42502         
42503         this.el.dom.value = '';
42504         if (this.hiddenEl) {
42505             this.hiddenEl.dom.value = '';
42506         }
42507         
42508     },
42509     getValue: function()
42510     {
42511         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42512     },
42513     setValue: function(v) // not a valid action - must use addItems..
42514     {
42515         
42516         this.reset();
42517          
42518         if (this.store.isLocal && (typeof(v) == 'string')) {
42519             // then we can use the store to find the values..
42520             // comma seperated at present.. this needs to allow JSON based encoding..
42521             this.hiddenEl.value  = v;
42522             var v_ar = [];
42523             Roo.each(v.split(','), function(k) {
42524                 Roo.log("CHECK " + this.valueField + ',' + k);
42525                 var li = this.store.query(this.valueField, k);
42526                 if (!li.length) {
42527                     return;
42528                 }
42529                 var add = {};
42530                 add[this.valueField] = k;
42531                 add[this.displayField] = li.item(0).data[this.displayField];
42532                 
42533                 this.addItem(add);
42534             }, this) 
42535              
42536         }
42537         if (typeof(v) == 'object' ) {
42538             // then let's assume it's an array of objects..
42539             Roo.each(v, function(l) {
42540                 this.addItem(l);
42541             }, this);
42542              
42543         }
42544         
42545         
42546     },
42547     setFromData: function(v)
42548     {
42549         // this recieves an object, if setValues is called.
42550         this.reset();
42551         this.el.dom.value = v[this.displayField];
42552         this.hiddenEl.dom.value = v[this.valueField];
42553         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42554             return;
42555         }
42556         var kv = v[this.valueField];
42557         var dv = v[this.displayField];
42558         kv = typeof(kv) != 'string' ? '' : kv;
42559         dv = typeof(dv) != 'string' ? '' : dv;
42560         
42561         
42562         var keys = kv.split(',');
42563         var display = dv.split(',');
42564         for (var i = 0 ; i < keys.length; i++) {
42565             
42566             add = {};
42567             add[this.valueField] = keys[i];
42568             add[this.displayField] = display[i];
42569             this.addItem(add);
42570         }
42571       
42572         
42573     },
42574     
42575     /**
42576      * Validates the combox array value
42577      * @return {Boolean} True if the value is valid, else false
42578      */
42579     validate : function(){
42580         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42581             this.clearInvalid();
42582             return true;
42583         }
42584         return false;
42585     },
42586     
42587     validateValue : function(value){
42588         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42589         
42590     },
42591     
42592     /*@
42593      * overide
42594      * 
42595      */
42596     isDirty : function() {
42597         if(this.disabled) {
42598             return false;
42599         }
42600         
42601         try {
42602             var d = Roo.decode(String(this.originalValue));
42603         } catch (e) {
42604             return String(this.getValue()) !== String(this.originalValue);
42605         }
42606         
42607         var originalValue = [];
42608         
42609         for (var i = 0; i < d.length; i++){
42610             originalValue.push(d[i][this.valueField]);
42611         }
42612         
42613         return String(this.getValue()) !== String(originalValue.join(','));
42614         
42615     }
42616     
42617 });
42618
42619
42620
42621 /**
42622  * @class Roo.form.ComboBoxArray.Item
42623  * @extends Roo.BoxComponent
42624  * A selected item in the list
42625  *  Fred [x]  Brian [x]  [Pick another |v]
42626  * 
42627  * @constructor
42628  * Create a new item.
42629  * @param {Object} config Configuration options
42630  */
42631  
42632 Roo.form.ComboBoxArray.Item = function(config) {
42633     config.id = Roo.id();
42634     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42635 }
42636
42637 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42638     data : {},
42639     cb: false,
42640     displayField : false,
42641     tipField : false,
42642     
42643     
42644     defaultAutoCreate : {
42645         tag: 'div',
42646         cls: 'x-cbarray-item',
42647         cn : [ 
42648             { tag: 'div' },
42649             {
42650                 tag: 'img',
42651                 width:16,
42652                 height : 16,
42653                 src : Roo.BLANK_IMAGE_URL ,
42654                 align: 'center'
42655             }
42656         ]
42657         
42658     },
42659     
42660  
42661     onRender : function(ct, position)
42662     {
42663         Roo.form.Field.superclass.onRender.call(this, ct, position);
42664         
42665         if(!this.el){
42666             var cfg = this.getAutoCreate();
42667             this.el = ct.createChild(cfg, position);
42668         }
42669         
42670         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42671         
42672         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42673             this.cb.renderer(this.data) :
42674             String.format('{0}',this.data[this.displayField]);
42675         
42676             
42677         this.el.child('div').dom.setAttribute('qtip',
42678                         String.format('{0}',this.data[this.tipField])
42679         );
42680         
42681         this.el.child('img').on('click', this.remove, this);
42682         
42683     },
42684    
42685     remove : function()
42686     {
42687         if(this.cb.disabled){
42688             return;
42689         }
42690         
42691         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42692             this.cb.items.remove(this);
42693             this.el.child('img').un('click', this.remove, this);
42694             this.el.remove();
42695             this.cb.updateHiddenEl();
42696
42697             this.cb.fireEvent('remove', this.cb, this);
42698         }
42699         
42700     }
42701 });/*
42702  * RooJS Library 1.1.1
42703  * Copyright(c) 2008-2011  Alan Knowles
42704  *
42705  * License - LGPL
42706  */
42707  
42708
42709 /**
42710  * @class Roo.form.ComboNested
42711  * @extends Roo.form.ComboBox
42712  * A combobox for that allows selection of nested items in a list,
42713  * eg.
42714  *
42715  *  Book
42716  *    -> red
42717  *    -> green
42718  *  Table
42719  *    -> square
42720  *      ->red
42721  *      ->green
42722  *    -> rectangle
42723  *      ->green
42724  *      
42725  * 
42726  * @constructor
42727  * Create a new ComboNested
42728  * @param {Object} config Configuration options
42729  */
42730 Roo.form.ComboNested = function(config){
42731     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42732     // should verify some data...
42733     // like
42734     // hiddenName = required..
42735     // displayField = required
42736     // valudField == required
42737     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42738     var _t = this;
42739     Roo.each(req, function(e) {
42740         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42741             throw "Roo.form.ComboNested : missing value for: " + e;
42742         }
42743     });
42744      
42745     
42746 };
42747
42748 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42749    
42750     /*
42751      * @config {Number} max Number of columns to show
42752      */
42753     
42754     maxColumns : 3,
42755    
42756     list : null, // the outermost div..
42757     innerLists : null, // the
42758     views : null,
42759     stores : null,
42760     // private
42761     onRender : function(ct, position)
42762     {
42763         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42764         
42765         if(this.hiddenName){
42766             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42767                     'before', true);
42768             this.hiddenField.value =
42769                 this.hiddenValue !== undefined ? this.hiddenValue :
42770                 this.value !== undefined ? this.value : '';
42771
42772             // prevent input submission
42773             this.el.dom.removeAttribute('name');
42774              
42775              
42776         }
42777         
42778         if(Roo.isGecko){
42779             this.el.dom.setAttribute('autocomplete', 'off');
42780         }
42781
42782         var cls = 'x-combo-list';
42783
42784         this.list = new Roo.Layer({
42785             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42786         });
42787
42788         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42789         this.list.setWidth(lw);
42790         this.list.swallowEvent('mousewheel');
42791         this.assetHeight = 0;
42792
42793         if(this.title){
42794             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42795             this.assetHeight += this.header.getHeight();
42796         }
42797         this.innerLists = [];
42798         this.views = [];
42799         this.stores = [];
42800         for (var i =0 ; i < this.maxColumns; i++) {
42801             this.onRenderList( cls, i);
42802         }
42803         
42804         // always needs footer, as we are going to have an 'OK' button.
42805         this.footer = this.list.createChild({cls:cls+'-ft'});
42806         this.pageTb = new Roo.Toolbar(this.footer);  
42807         var _this = this;
42808         this.pageTb.add(  {
42809             
42810             text: 'Done',
42811             handler: function()
42812             {
42813                 _this.collapse();
42814             }
42815         });
42816         
42817         if ( this.allowBlank && !this.disableClear) {
42818             
42819             this.pageTb.add(new Roo.Toolbar.Fill(), {
42820                 cls: 'x-btn-icon x-btn-clear',
42821                 text: '&#160;',
42822                 handler: function()
42823                 {
42824                     _this.collapse();
42825                     _this.clearValue();
42826                     _this.onSelect(false, -1);
42827                 }
42828             });
42829         }
42830         if (this.footer) {
42831             this.assetHeight += this.footer.getHeight();
42832         }
42833         
42834     },
42835     onRenderList : function (  cls, i)
42836     {
42837         
42838         var lw = Math.floor(
42839                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
42840         );
42841         
42842         this.list.setWidth(lw); // default to '1'
42843
42844         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42845         //il.on('mouseover', this.onViewOver, this, { list:  i });
42846         //il.on('mousemove', this.onViewMove, this, { list:  i });
42847         il.setWidth(lw);
42848         il.setStyle({ 'overflow-x' : 'hidden'});
42849
42850         if(!this.tpl){
42851             this.tpl = new Roo.Template({
42852                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42853                 isEmpty: function (value, allValues) {
42854                     //Roo.log(value);
42855                     var dl = typeof(value.data) != 'undefined' ? value.data.length : value.length; ///json is a nested response..
42856                     return dl ? 'has-children' : 'no-children'
42857                 }
42858             });
42859         }
42860         
42861         var store  = this.store;
42862         if (i > 0) {
42863             store  = new Roo.data.SimpleStore({
42864                 //fields : this.store.reader.meta.fields,
42865                 reader : this.store.reader,
42866                 data : [ ]
42867             });
42868         }
42869         this.stores[i]  = store;
42870                 
42871         
42872         
42873         var view = this.views[i] = new Roo.View(
42874             il,
42875             this.tpl,
42876             {
42877                 singleSelect:true,
42878                 store: store,
42879                 selectedClass: this.selectedClass
42880             }
42881         );
42882         view.getEl().setWidth(lw);
42883         view.getEl().setStyle({
42884             position: i < 1 ? 'relative' : 'absolute',
42885             top: 0,
42886             left: (i * lw ) + 'px',
42887             display : i > 0 ? 'none' : 'block'
42888         });
42889         view.on('selectionchange', this.onSelectChange, this, {list : i });
42890         view.on('dblclick', this.onDoubleClick, this, {list : i });
42891         //view.on('click', this.onViewClick, this, { list : i });
42892
42893         store.on('beforeload', this.onBeforeLoad, this);
42894         store.on('load',  this.onLoad, this, { list  : i});
42895         store.on('loadexception', this.onLoadException, this);
42896
42897         // hide the other vies..
42898         
42899         
42900         
42901     },
42902     onResize : function()  {},
42903     
42904     restrictHeight : function()
42905     {
42906         var mh = 0;
42907         Roo.each(this.innerLists, function(il,i) {
42908             var el = this.views[i].getEl();
42909             el.dom.style.height = '';
42910             var inner = el.dom;
42911             var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
42912             // only adjust heights on other ones..
42913             if (i < 1) {
42914                 
42915                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42916                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42917                 mh = Math.max(el.getHeight(), mh);
42918             }
42919             
42920             
42921         }, this);
42922         
42923         this.list.beginUpdate();
42924         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42925         this.list.alignTo(this.el, this.listAlign);
42926         this.list.endUpdate();
42927         
42928     },
42929      
42930     
42931     // -- store handlers..
42932     // private
42933     onBeforeLoad : function()
42934     {
42935         if(!this.hasFocus){
42936             return;
42937         }
42938         this.innerLists[0].update(this.loadingText ?
42939                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
42940         this.restrictHeight();
42941         this.selectedIndex = -1;
42942     },
42943     // private
42944     onLoad : function(a,b,c,d)
42945     {
42946         
42947         if(!this.hasFocus){
42948             return;
42949         }
42950         
42951         if(this.store.getCount() > 0) {
42952             this.expand();
42953             this.restrictHeight();   
42954         } else {
42955             this.onEmptyResults();
42956         }
42957         /*
42958         this.stores[1].loadData([]);
42959         this.stores[2].loadData([]);
42960         this.views
42961         */    
42962     
42963         //this.el.focus();
42964     },
42965     
42966     
42967     // private
42968     onLoadException : function()
42969     {
42970         this.collapse();
42971         Roo.log(this.store.reader.jsonData);
42972         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
42973             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
42974         }
42975         
42976         
42977     } ,
42978      
42979      
42980
42981     onSelectChange : function (view, sels, opts )
42982     {
42983         var ix = view.getSelectedIndexes();
42984         
42985         
42986         if (opts.list > this.maxColumns - 2) {
42987              
42988             this.setFromData(ix.length ? view.store.getAt(ix[0]).data : {});
42989             return;
42990         }
42991         
42992         if (!ix.length) {
42993             this.setFromData({});
42994             var str = this.stores[opts.list+1];
42995             str.removeAll();
42996             return;
42997         }
42998         
42999         var rec = view.store.getAt(ix[0]);
43000         this.setFromData(rec.data);
43001         
43002         var lw = Math.floor(
43003                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
43004         );
43005         var data =  typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
43006         var dl = typeof(data.data) != 'undefined' ? data.total : data.length; ///json is a nested response..
43007         this.stores[opts.list+1].loadData( data );
43008         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
43009         this.views[opts.list+1].getEl().setStyle({ display : dl ? 'block' : 'none' });
43010         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
43011         this.list.setWidth(lw * (opts.list + (dl ? 2 : 1))); 
43012     },
43013     onDoubleClick : function()
43014     {
43015         this.collapse(); //??
43016     },
43017     
43018      
43019     
43020     findRecord : function (prop,value)
43021     {
43022         return this.findRecordInStore(this.store, prop,value);
43023     },
43024     
43025      // private
43026     findRecordInStore : function(store, prop, value)
43027     {
43028         var cstore = new Roo.data.SimpleStore({
43029             //fields : this.store.reader.meta.fields, // we need array reader.. for
43030             reader : this.store.reader,
43031             data : [ ]
43032         });
43033         var _this = this;
43034         var record  = false;
43035         if(store.getCount() > 0){
43036            store.each(function(r){
43037                 if(r.data[prop] == value){
43038                     record = r;
43039                     return false;
43040                 }
43041                 if (r.data.cn && r.data.cn.length) {
43042                     cstore.loadData( r.data.cn);
43043                     var cret = _this.findRecordInStore(cstore, prop, value);
43044                     if (cret !== false) {
43045                         record = cret;
43046                         return false;
43047                     }
43048                 }
43049                 
43050                 return true;
43051             });
43052         }
43053         return record;
43054     }
43055     
43056     
43057     
43058     
43059 });/*
43060  * Based on:
43061  * Ext JS Library 1.1.1
43062  * Copyright(c) 2006-2007, Ext JS, LLC.
43063  *
43064  * Originally Released Under LGPL - original licence link has changed is not relivant.
43065  *
43066  * Fork - LGPL
43067  * <script type="text/javascript">
43068  */
43069 /**
43070  * @class Roo.form.Checkbox
43071  * @extends Roo.form.Field
43072  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43073  * @constructor
43074  * Creates a new Checkbox
43075  * @param {Object} config Configuration options
43076  */
43077 Roo.form.Checkbox = function(config){
43078     Roo.form.Checkbox.superclass.constructor.call(this, config);
43079     this.addEvents({
43080         /**
43081          * @event check
43082          * Fires when the checkbox is checked or unchecked.
43083              * @param {Roo.form.Checkbox} this This checkbox
43084              * @param {Boolean} checked The new checked value
43085              */
43086         check : true
43087     });
43088 };
43089
43090 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43091     /**
43092      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43093      */
43094     focusClass : undefined,
43095     /**
43096      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43097      */
43098     fieldClass: "x-form-field",
43099     /**
43100      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43101      */
43102     checked: false,
43103     /**
43104      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43105      * {tag: "input", type: "checkbox", autocomplete: "off"})
43106      */
43107     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43108     /**
43109      * @cfg {String} boxLabel The text that appears beside the checkbox
43110      */
43111     boxLabel : "",
43112     /**
43113      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43114      */  
43115     inputValue : '1',
43116     /**
43117      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43118      */
43119      valueOff: '0', // value when not checked..
43120
43121     actionMode : 'viewEl', 
43122     //
43123     // private
43124     itemCls : 'x-menu-check-item x-form-item',
43125     groupClass : 'x-menu-group-item',
43126     inputType : 'hidden',
43127     
43128     
43129     inSetChecked: false, // check that we are not calling self...
43130     
43131     inputElement: false, // real input element?
43132     basedOn: false, // ????
43133     
43134     isFormField: true, // not sure where this is needed!!!!
43135
43136     onResize : function(){
43137         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43138         if(!this.boxLabel){
43139             this.el.alignTo(this.wrap, 'c-c');
43140         }
43141     },
43142
43143     initEvents : function(){
43144         Roo.form.Checkbox.superclass.initEvents.call(this);
43145         this.el.on("click", this.onClick,  this);
43146         this.el.on("change", this.onClick,  this);
43147     },
43148
43149
43150     getResizeEl : function(){
43151         return this.wrap;
43152     },
43153
43154     getPositionEl : function(){
43155         return this.wrap;
43156     },
43157
43158     // private
43159     onRender : function(ct, position){
43160         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43161         /*
43162         if(this.inputValue !== undefined){
43163             this.el.dom.value = this.inputValue;
43164         }
43165         */
43166         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43167         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43168         var viewEl = this.wrap.createChild({ 
43169             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43170         this.viewEl = viewEl;   
43171         this.wrap.on('click', this.onClick,  this); 
43172         
43173         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43174         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43175         
43176         
43177         
43178         if(this.boxLabel){
43179             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43180         //    viewEl.on('click', this.onClick,  this); 
43181         }
43182         //if(this.checked){
43183             this.setChecked(this.checked);
43184         //}else{
43185             //this.checked = this.el.dom;
43186         //}
43187
43188     },
43189
43190     // private
43191     initValue : Roo.emptyFn,
43192
43193     /**
43194      * Returns the checked state of the checkbox.
43195      * @return {Boolean} True if checked, else false
43196      */
43197     getValue : function(){
43198         if(this.el){
43199             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43200         }
43201         return this.valueOff;
43202         
43203     },
43204
43205         // private
43206     onClick : function(){ 
43207         if (this.disabled) {
43208             return;
43209         }
43210         this.setChecked(!this.checked);
43211
43212         //if(this.el.dom.checked != this.checked){
43213         //    this.setValue(this.el.dom.checked);
43214        // }
43215     },
43216
43217     /**
43218      * Sets the checked state of the checkbox.
43219      * On is always based on a string comparison between inputValue and the param.
43220      * @param {Boolean/String} value - the value to set 
43221      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43222      */
43223     setValue : function(v,suppressEvent){
43224         
43225         
43226         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43227         //if(this.el && this.el.dom){
43228         //    this.el.dom.checked = this.checked;
43229         //    this.el.dom.defaultChecked = this.checked;
43230         //}
43231         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43232         //this.fireEvent("check", this, this.checked);
43233     },
43234     // private..
43235     setChecked : function(state,suppressEvent)
43236     {
43237         if (this.inSetChecked) {
43238             this.checked = state;
43239             return;
43240         }
43241         
43242     
43243         if(this.wrap){
43244             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43245         }
43246         this.checked = state;
43247         if(suppressEvent !== true){
43248             this.fireEvent('check', this, state);
43249         }
43250         this.inSetChecked = true;
43251         this.el.dom.value = state ? this.inputValue : this.valueOff;
43252         this.inSetChecked = false;
43253         
43254     },
43255     // handle setting of hidden value by some other method!!?!?
43256     setFromHidden: function()
43257     {
43258         if(!this.el){
43259             return;
43260         }
43261         //console.log("SET FROM HIDDEN");
43262         //alert('setFrom hidden');
43263         this.setValue(this.el.dom.value);
43264     },
43265     
43266     onDestroy : function()
43267     {
43268         if(this.viewEl){
43269             Roo.get(this.viewEl).remove();
43270         }
43271          
43272         Roo.form.Checkbox.superclass.onDestroy.call(this);
43273     },
43274     
43275     setBoxLabel : function(str)
43276     {
43277         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43278     }
43279
43280 });/*
43281  * Based on:
43282  * Ext JS Library 1.1.1
43283  * Copyright(c) 2006-2007, Ext JS, LLC.
43284  *
43285  * Originally Released Under LGPL - original licence link has changed is not relivant.
43286  *
43287  * Fork - LGPL
43288  * <script type="text/javascript">
43289  */
43290  
43291 /**
43292  * @class Roo.form.Radio
43293  * @extends Roo.form.Checkbox
43294  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43295  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43296  * @constructor
43297  * Creates a new Radio
43298  * @param {Object} config Configuration options
43299  */
43300 Roo.form.Radio = function(){
43301     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43302 };
43303 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43304     inputType: 'radio',
43305
43306     /**
43307      * If this radio is part of a group, it will return the selected value
43308      * @return {String}
43309      */
43310     getGroupValue : function(){
43311         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43312     },
43313     
43314     
43315     onRender : function(ct, position){
43316         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43317         
43318         if(this.inputValue !== undefined){
43319             this.el.dom.value = this.inputValue;
43320         }
43321          
43322         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43323         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43324         //var viewEl = this.wrap.createChild({ 
43325         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43326         //this.viewEl = viewEl;   
43327         //this.wrap.on('click', this.onClick,  this); 
43328         
43329         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43330         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43331         
43332         
43333         
43334         if(this.boxLabel){
43335             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43336         //    viewEl.on('click', this.onClick,  this); 
43337         }
43338          if(this.checked){
43339             this.el.dom.checked =   'checked' ;
43340         }
43341          
43342     } 
43343     
43344     
43345 });//<script type="text/javascript">
43346
43347 /*
43348  * Based  Ext JS Library 1.1.1
43349  * Copyright(c) 2006-2007, Ext JS, LLC.
43350  * LGPL
43351  *
43352  */
43353  
43354 /**
43355  * @class Roo.HtmlEditorCore
43356  * @extends Roo.Component
43357  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43358  *
43359  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43360  */
43361
43362 Roo.HtmlEditorCore = function(config){
43363     
43364     
43365     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43366     
43367     
43368     this.addEvents({
43369         /**
43370          * @event initialize
43371          * Fires when the editor is fully initialized (including the iframe)
43372          * @param {Roo.HtmlEditorCore} this
43373          */
43374         initialize: true,
43375         /**
43376          * @event activate
43377          * Fires when the editor is first receives the focus. Any insertion must wait
43378          * until after this event.
43379          * @param {Roo.HtmlEditorCore} this
43380          */
43381         activate: true,
43382          /**
43383          * @event beforesync
43384          * Fires before the textarea is updated with content from the editor iframe. Return false
43385          * to cancel the sync.
43386          * @param {Roo.HtmlEditorCore} this
43387          * @param {String} html
43388          */
43389         beforesync: true,
43390          /**
43391          * @event beforepush
43392          * Fires before the iframe editor is updated with content from the textarea. Return false
43393          * to cancel the push.
43394          * @param {Roo.HtmlEditorCore} this
43395          * @param {String} html
43396          */
43397         beforepush: true,
43398          /**
43399          * @event sync
43400          * Fires when the textarea is updated with content from the editor iframe.
43401          * @param {Roo.HtmlEditorCore} this
43402          * @param {String} html
43403          */
43404         sync: true,
43405          /**
43406          * @event push
43407          * Fires when the iframe editor is updated with content from the textarea.
43408          * @param {Roo.HtmlEditorCore} this
43409          * @param {String} html
43410          */
43411         push: true,
43412         
43413         /**
43414          * @event editorevent
43415          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43416          * @param {Roo.HtmlEditorCore} this
43417          */
43418         editorevent: true
43419         
43420     });
43421     
43422     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43423     
43424     // defaults : white / black...
43425     this.applyBlacklists();
43426     
43427     
43428     
43429 };
43430
43431
43432 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43433
43434
43435      /**
43436      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43437      */
43438     
43439     owner : false,
43440     
43441      /**
43442      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43443      *                        Roo.resizable.
43444      */
43445     resizable : false,
43446      /**
43447      * @cfg {Number} height (in pixels)
43448      */   
43449     height: 300,
43450    /**
43451      * @cfg {Number} width (in pixels)
43452      */   
43453     width: 500,
43454     
43455     /**
43456      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43457      * 
43458      */
43459     stylesheets: false,
43460     
43461     // id of frame..
43462     frameId: false,
43463     
43464     // private properties
43465     validationEvent : false,
43466     deferHeight: true,
43467     initialized : false,
43468     activated : false,
43469     sourceEditMode : false,
43470     onFocus : Roo.emptyFn,
43471     iframePad:3,
43472     hideMode:'offsets',
43473     
43474     clearUp: true,
43475     
43476     // blacklist + whitelisted elements..
43477     black: false,
43478     white: false,
43479      
43480     bodyCls : '',
43481
43482     /**
43483      * Protected method that will not generally be called directly. It
43484      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43485      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43486      */
43487     getDocMarkup : function(){
43488         // body styles..
43489         var st = '';
43490         
43491         // inherit styels from page...?? 
43492         if (this.stylesheets === false) {
43493             
43494             Roo.get(document.head).select('style').each(function(node) {
43495                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43496             });
43497             
43498             Roo.get(document.head).select('link').each(function(node) { 
43499                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43500             });
43501             
43502         } else if (!this.stylesheets.length) {
43503                 // simple..
43504                 st = '<style type="text/css">' +
43505                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43506                    '</style>';
43507         } else { 
43508             st = '<style type="text/css">' +
43509                     this.stylesheets +
43510                 '</style>';
43511         }
43512         
43513         st +=  '<style type="text/css">' +
43514             'IMG { cursor: pointer } ' +
43515         '</style>';
43516
43517         var cls = 'roo-htmleditor-body';
43518         
43519         if(this.bodyCls.length){
43520             cls += ' ' + this.bodyCls;
43521         }
43522         
43523         return '<html><head>' + st  +
43524             //<style type="text/css">' +
43525             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43526             //'</style>' +
43527             ' </head><body class="' +  cls + '"></body></html>';
43528     },
43529
43530     // private
43531     onRender : function(ct, position)
43532     {
43533         var _t = this;
43534         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43535         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43536         
43537         
43538         this.el.dom.style.border = '0 none';
43539         this.el.dom.setAttribute('tabIndex', -1);
43540         this.el.addClass('x-hidden hide');
43541         
43542         
43543         
43544         if(Roo.isIE){ // fix IE 1px bogus margin
43545             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43546         }
43547        
43548         
43549         this.frameId = Roo.id();
43550         
43551          
43552         
43553         var iframe = this.owner.wrap.createChild({
43554             tag: 'iframe',
43555             cls: 'form-control', // bootstrap..
43556             id: this.frameId,
43557             name: this.frameId,
43558             frameBorder : 'no',
43559             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43560         }, this.el
43561         );
43562         
43563         
43564         this.iframe = iframe.dom;
43565
43566          this.assignDocWin();
43567         
43568         this.doc.designMode = 'on';
43569        
43570         this.doc.open();
43571         this.doc.write(this.getDocMarkup());
43572         this.doc.close();
43573
43574         
43575         var task = { // must defer to wait for browser to be ready
43576             run : function(){
43577                 //console.log("run task?" + this.doc.readyState);
43578                 this.assignDocWin();
43579                 if(this.doc.body || this.doc.readyState == 'complete'){
43580                     try {
43581                         this.doc.designMode="on";
43582                     } catch (e) {
43583                         return;
43584                     }
43585                     Roo.TaskMgr.stop(task);
43586                     this.initEditor.defer(10, this);
43587                 }
43588             },
43589             interval : 10,
43590             duration: 10000,
43591             scope: this
43592         };
43593         Roo.TaskMgr.start(task);
43594
43595     },
43596
43597     // private
43598     onResize : function(w, h)
43599     {
43600          Roo.log('resize: ' +w + ',' + h );
43601         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43602         if(!this.iframe){
43603             return;
43604         }
43605         if(typeof w == 'number'){
43606             
43607             this.iframe.style.width = w + 'px';
43608         }
43609         if(typeof h == 'number'){
43610             
43611             this.iframe.style.height = h + 'px';
43612             if(this.doc){
43613                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43614             }
43615         }
43616         
43617     },
43618
43619     /**
43620      * Toggles the editor between standard and source edit mode.
43621      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43622      */
43623     toggleSourceEdit : function(sourceEditMode){
43624         
43625         this.sourceEditMode = sourceEditMode === true;
43626         
43627         if(this.sourceEditMode){
43628  
43629             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43630             
43631         }else{
43632             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43633             //this.iframe.className = '';
43634             this.deferFocus();
43635         }
43636         //this.setSize(this.owner.wrap.getSize());
43637         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43638     },
43639
43640     
43641   
43642
43643     /**
43644      * Protected method that will not generally be called directly. If you need/want
43645      * custom HTML cleanup, this is the method you should override.
43646      * @param {String} html The HTML to be cleaned
43647      * return {String} The cleaned HTML
43648      */
43649     cleanHtml : function(html){
43650         html = String(html);
43651         if(html.length > 5){
43652             if(Roo.isSafari){ // strip safari nonsense
43653                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43654             }
43655         }
43656         if(html == '&nbsp;'){
43657             html = '';
43658         }
43659         return html;
43660     },
43661
43662     /**
43663      * HTML Editor -> Textarea
43664      * Protected method that will not generally be called directly. Syncs the contents
43665      * of the editor iframe with the textarea.
43666      */
43667     syncValue : function(){
43668         if(this.initialized){
43669             var bd = (this.doc.body || this.doc.documentElement);
43670             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43671             var html = bd.innerHTML;
43672             if(Roo.isSafari){
43673                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43674                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43675                 if(m && m[1]){
43676                     html = '<div style="'+m[0]+'">' + html + '</div>';
43677                 }
43678             }
43679             html = this.cleanHtml(html);
43680             // fix up the special chars.. normaly like back quotes in word...
43681             // however we do not want to do this with chinese..
43682             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43683                 
43684                 var cc = match.charCodeAt();
43685
43686                 // Get the character value, handling surrogate pairs
43687                 if (match.length == 2) {
43688                     // It's a surrogate pair, calculate the Unicode code point
43689                     var high = match.charCodeAt(0) - 0xD800;
43690                     var low  = match.charCodeAt(1) - 0xDC00;
43691                     cc = (high * 0x400) + low + 0x10000;
43692                 }  else if (
43693                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43694                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43695                     (cc >= 0xf900 && cc < 0xfb00 )
43696                 ) {
43697                         return match;
43698                 }  
43699          
43700                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43701                 return "&#" + cc + ";";
43702                 
43703                 
43704             });
43705             
43706             
43707              
43708             if(this.owner.fireEvent('beforesync', this, html) !== false){
43709                 this.el.dom.value = html;
43710                 this.owner.fireEvent('sync', this, html);
43711             }
43712         }
43713     },
43714
43715     /**
43716      * Protected method that will not generally be called directly. Pushes the value of the textarea
43717      * into the iframe editor.
43718      */
43719     pushValue : function(){
43720         if(this.initialized){
43721             var v = this.el.dom.value.trim();
43722             
43723 //            if(v.length < 1){
43724 //                v = '&#160;';
43725 //            }
43726             
43727             if(this.owner.fireEvent('beforepush', this, v) !== false){
43728                 var d = (this.doc.body || this.doc.documentElement);
43729                 d.innerHTML = v;
43730                 this.cleanUpPaste();
43731                 this.el.dom.value = d.innerHTML;
43732                 this.owner.fireEvent('push', this, v);
43733             }
43734         }
43735     },
43736
43737     // private
43738     deferFocus : function(){
43739         this.focus.defer(10, this);
43740     },
43741
43742     // doc'ed in Field
43743     focus : function(){
43744         if(this.win && !this.sourceEditMode){
43745             this.win.focus();
43746         }else{
43747             this.el.focus();
43748         }
43749     },
43750     
43751     assignDocWin: function()
43752     {
43753         var iframe = this.iframe;
43754         
43755          if(Roo.isIE){
43756             this.doc = iframe.contentWindow.document;
43757             this.win = iframe.contentWindow;
43758         } else {
43759 //            if (!Roo.get(this.frameId)) {
43760 //                return;
43761 //            }
43762 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43763 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43764             
43765             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43766                 return;
43767             }
43768             
43769             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43770             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43771         }
43772     },
43773     
43774     // private
43775     initEditor : function(){
43776         //console.log("INIT EDITOR");
43777         this.assignDocWin();
43778         
43779         
43780         
43781         this.doc.designMode="on";
43782         this.doc.open();
43783         this.doc.write(this.getDocMarkup());
43784         this.doc.close();
43785         
43786         var dbody = (this.doc.body || this.doc.documentElement);
43787         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43788         // this copies styles from the containing element into thsi one..
43789         // not sure why we need all of this..
43790         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43791         
43792         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43793         //ss['background-attachment'] = 'fixed'; // w3c
43794         dbody.bgProperties = 'fixed'; // ie
43795         //Roo.DomHelper.applyStyles(dbody, ss);
43796         Roo.EventManager.on(this.doc, {
43797             //'mousedown': this.onEditorEvent,
43798             'mouseup': this.onEditorEvent,
43799             'dblclick': this.onEditorEvent,
43800             'click': this.onEditorEvent,
43801             'keyup': this.onEditorEvent,
43802             buffer:100,
43803             scope: this
43804         });
43805         if(Roo.isGecko){
43806             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43807         }
43808         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43809             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43810         }
43811         this.initialized = true;
43812
43813         this.owner.fireEvent('initialize', this);
43814         this.pushValue();
43815     },
43816
43817     // private
43818     onDestroy : function(){
43819         
43820         
43821         
43822         if(this.rendered){
43823             
43824             //for (var i =0; i < this.toolbars.length;i++) {
43825             //    // fixme - ask toolbars for heights?
43826             //    this.toolbars[i].onDestroy();
43827            // }
43828             
43829             //this.wrap.dom.innerHTML = '';
43830             //this.wrap.remove();
43831         }
43832     },
43833
43834     // private
43835     onFirstFocus : function(){
43836         
43837         this.assignDocWin();
43838         
43839         
43840         this.activated = true;
43841          
43842     
43843         if(Roo.isGecko){ // prevent silly gecko errors
43844             this.win.focus();
43845             var s = this.win.getSelection();
43846             if(!s.focusNode || s.focusNode.nodeType != 3){
43847                 var r = s.getRangeAt(0);
43848                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43849                 r.collapse(true);
43850                 this.deferFocus();
43851             }
43852             try{
43853                 this.execCmd('useCSS', true);
43854                 this.execCmd('styleWithCSS', false);
43855             }catch(e){}
43856         }
43857         this.owner.fireEvent('activate', this);
43858     },
43859
43860     // private
43861     adjustFont: function(btn){
43862         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43863         //if(Roo.isSafari){ // safari
43864         //    adjust *= 2;
43865        // }
43866         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43867         if(Roo.isSafari){ // safari
43868             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43869             v =  (v < 10) ? 10 : v;
43870             v =  (v > 48) ? 48 : v;
43871             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43872             
43873         }
43874         
43875         
43876         v = Math.max(1, v+adjust);
43877         
43878         this.execCmd('FontSize', v  );
43879     },
43880
43881     onEditorEvent : function(e)
43882     {
43883         this.owner.fireEvent('editorevent', this, e);
43884       //  this.updateToolbar();
43885         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43886     },
43887
43888     insertTag : function(tg)
43889     {
43890         // could be a bit smarter... -> wrap the current selected tRoo..
43891         if (tg.toLowerCase() == 'span' ||
43892             tg.toLowerCase() == 'code' ||
43893             tg.toLowerCase() == 'sup' ||
43894             tg.toLowerCase() == 'sub' 
43895             ) {
43896             
43897             range = this.createRange(this.getSelection());
43898             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43899             wrappingNode.appendChild(range.extractContents());
43900             range.insertNode(wrappingNode);
43901
43902             return;
43903             
43904             
43905             
43906         }
43907         this.execCmd("formatblock",   tg);
43908         
43909     },
43910     
43911     insertText : function(txt)
43912     {
43913         
43914         
43915         var range = this.createRange();
43916         range.deleteContents();
43917                //alert(Sender.getAttribute('label'));
43918                
43919         range.insertNode(this.doc.createTextNode(txt));
43920     } ,
43921     
43922      
43923
43924     /**
43925      * Executes a Midas editor command on the editor document and performs necessary focus and
43926      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43927      * @param {String} cmd The Midas command
43928      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43929      */
43930     relayCmd : function(cmd, value){
43931         this.win.focus();
43932         this.execCmd(cmd, value);
43933         this.owner.fireEvent('editorevent', this);
43934         //this.updateToolbar();
43935         this.owner.deferFocus();
43936     },
43937
43938     /**
43939      * Executes a Midas editor command directly on the editor document.
43940      * For visual commands, you should use {@link #relayCmd} instead.
43941      * <b>This should only be called after the editor is initialized.</b>
43942      * @param {String} cmd The Midas command
43943      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43944      */
43945     execCmd : function(cmd, value){
43946         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43947         this.syncValue();
43948     },
43949  
43950  
43951    
43952     /**
43953      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43954      * to insert tRoo.
43955      * @param {String} text | dom node.. 
43956      */
43957     insertAtCursor : function(text)
43958     {
43959         
43960         if(!this.activated){
43961             return;
43962         }
43963         /*
43964         if(Roo.isIE){
43965             this.win.focus();
43966             var r = this.doc.selection.createRange();
43967             if(r){
43968                 r.collapse(true);
43969                 r.pasteHTML(text);
43970                 this.syncValue();
43971                 this.deferFocus();
43972             
43973             }
43974             return;
43975         }
43976         */
43977         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43978             this.win.focus();
43979             
43980             
43981             // from jquery ui (MIT licenced)
43982             var range, node;
43983             var win = this.win;
43984             
43985             if (win.getSelection && win.getSelection().getRangeAt) {
43986                 range = win.getSelection().getRangeAt(0);
43987                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43988                 range.insertNode(node);
43989             } else if (win.document.selection && win.document.selection.createRange) {
43990                 // no firefox support
43991                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43992                 win.document.selection.createRange().pasteHTML(txt);
43993             } else {
43994                 // no firefox support
43995                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43996                 this.execCmd('InsertHTML', txt);
43997             } 
43998             
43999             this.syncValue();
44000             
44001             this.deferFocus();
44002         }
44003     },
44004  // private
44005     mozKeyPress : function(e){
44006         if(e.ctrlKey){
44007             var c = e.getCharCode(), cmd;
44008           
44009             if(c > 0){
44010                 c = String.fromCharCode(c).toLowerCase();
44011                 switch(c){
44012                     case 'b':
44013                         cmd = 'bold';
44014                         break;
44015                     case 'i':
44016                         cmd = 'italic';
44017                         break;
44018                     
44019                     case 'u':
44020                         cmd = 'underline';
44021                         break;
44022                     
44023                     case 'v':
44024                         this.cleanUpPaste.defer(100, this);
44025                         return;
44026                         
44027                 }
44028                 if(cmd){
44029                     this.win.focus();
44030                     this.execCmd(cmd);
44031                     this.deferFocus();
44032                     e.preventDefault();
44033                 }
44034                 
44035             }
44036         }
44037     },
44038
44039     // private
44040     fixKeys : function(){ // load time branching for fastest keydown performance
44041         if(Roo.isIE){
44042             return function(e){
44043                 var k = e.getKey(), r;
44044                 if(k == e.TAB){
44045                     e.stopEvent();
44046                     r = this.doc.selection.createRange();
44047                     if(r){
44048                         r.collapse(true);
44049                         r.pasteHTML('&#160;&#160;&#160;&#160;');
44050                         this.deferFocus();
44051                     }
44052                     return;
44053                 }
44054                 
44055                 if(k == e.ENTER){
44056                     r = this.doc.selection.createRange();
44057                     if(r){
44058                         var target = r.parentElement();
44059                         if(!target || target.tagName.toLowerCase() != 'li'){
44060                             e.stopEvent();
44061                             r.pasteHTML('<br />');
44062                             r.collapse(false);
44063                             r.select();
44064                         }
44065                     }
44066                 }
44067                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44068                     this.cleanUpPaste.defer(100, this);
44069                     return;
44070                 }
44071                 
44072                 
44073             };
44074         }else if(Roo.isOpera){
44075             return function(e){
44076                 var k = e.getKey();
44077                 if(k == e.TAB){
44078                     e.stopEvent();
44079                     this.win.focus();
44080                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44081                     this.deferFocus();
44082                 }
44083                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44084                     this.cleanUpPaste.defer(100, this);
44085                     return;
44086                 }
44087                 
44088             };
44089         }else if(Roo.isSafari){
44090             return function(e){
44091                 var k = e.getKey();
44092                 
44093                 if(k == e.TAB){
44094                     e.stopEvent();
44095                     this.execCmd('InsertText','\t');
44096                     this.deferFocus();
44097                     return;
44098                 }
44099                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44100                     this.cleanUpPaste.defer(100, this);
44101                     return;
44102                 }
44103                 
44104              };
44105         }
44106     }(),
44107     
44108     getAllAncestors: function()
44109     {
44110         var p = this.getSelectedNode();
44111         var a = [];
44112         if (!p) {
44113             a.push(p); // push blank onto stack..
44114             p = this.getParentElement();
44115         }
44116         
44117         
44118         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44119             a.push(p);
44120             p = p.parentNode;
44121         }
44122         a.push(this.doc.body);
44123         return a;
44124     },
44125     lastSel : false,
44126     lastSelNode : false,
44127     
44128     
44129     getSelection : function() 
44130     {
44131         this.assignDocWin();
44132         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44133     },
44134     
44135     getSelectedNode: function() 
44136     {
44137         // this may only work on Gecko!!!
44138         
44139         // should we cache this!!!!
44140         
44141         
44142         
44143          
44144         var range = this.createRange(this.getSelection()).cloneRange();
44145         
44146         if (Roo.isIE) {
44147             var parent = range.parentElement();
44148             while (true) {
44149                 var testRange = range.duplicate();
44150                 testRange.moveToElementText(parent);
44151                 if (testRange.inRange(range)) {
44152                     break;
44153                 }
44154                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44155                     break;
44156                 }
44157                 parent = parent.parentElement;
44158             }
44159             return parent;
44160         }
44161         
44162         // is ancestor a text element.
44163         var ac =  range.commonAncestorContainer;
44164         if (ac.nodeType == 3) {
44165             ac = ac.parentNode;
44166         }
44167         
44168         var ar = ac.childNodes;
44169          
44170         var nodes = [];
44171         var other_nodes = [];
44172         var has_other_nodes = false;
44173         for (var i=0;i<ar.length;i++) {
44174             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44175                 continue;
44176             }
44177             // fullly contained node.
44178             
44179             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44180                 nodes.push(ar[i]);
44181                 continue;
44182             }
44183             
44184             // probably selected..
44185             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44186                 other_nodes.push(ar[i]);
44187                 continue;
44188             }
44189             // outer..
44190             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44191                 continue;
44192             }
44193             
44194             
44195             has_other_nodes = true;
44196         }
44197         if (!nodes.length && other_nodes.length) {
44198             nodes= other_nodes;
44199         }
44200         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44201             return false;
44202         }
44203         
44204         return nodes[0];
44205     },
44206     createRange: function(sel)
44207     {
44208         // this has strange effects when using with 
44209         // top toolbar - not sure if it's a great idea.
44210         //this.editor.contentWindow.focus();
44211         if (typeof sel != "undefined") {
44212             try {
44213                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44214             } catch(e) {
44215                 return this.doc.createRange();
44216             }
44217         } else {
44218             return this.doc.createRange();
44219         }
44220     },
44221     getParentElement: function()
44222     {
44223         
44224         this.assignDocWin();
44225         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44226         
44227         var range = this.createRange(sel);
44228          
44229         try {
44230             var p = range.commonAncestorContainer;
44231             while (p.nodeType == 3) { // text node
44232                 p = p.parentNode;
44233             }
44234             return p;
44235         } catch (e) {
44236             return null;
44237         }
44238     
44239     },
44240     /***
44241      *
44242      * Range intersection.. the hard stuff...
44243      *  '-1' = before
44244      *  '0' = hits..
44245      *  '1' = after.
44246      *         [ -- selected range --- ]
44247      *   [fail]                        [fail]
44248      *
44249      *    basically..
44250      *      if end is before start or  hits it. fail.
44251      *      if start is after end or hits it fail.
44252      *
44253      *   if either hits (but other is outside. - then it's not 
44254      *   
44255      *    
44256      **/
44257     
44258     
44259     // @see http://www.thismuchiknow.co.uk/?p=64.
44260     rangeIntersectsNode : function(range, node)
44261     {
44262         var nodeRange = node.ownerDocument.createRange();
44263         try {
44264             nodeRange.selectNode(node);
44265         } catch (e) {
44266             nodeRange.selectNodeContents(node);
44267         }
44268     
44269         var rangeStartRange = range.cloneRange();
44270         rangeStartRange.collapse(true);
44271     
44272         var rangeEndRange = range.cloneRange();
44273         rangeEndRange.collapse(false);
44274     
44275         var nodeStartRange = nodeRange.cloneRange();
44276         nodeStartRange.collapse(true);
44277     
44278         var nodeEndRange = nodeRange.cloneRange();
44279         nodeEndRange.collapse(false);
44280     
44281         return rangeStartRange.compareBoundaryPoints(
44282                  Range.START_TO_START, nodeEndRange) == -1 &&
44283                rangeEndRange.compareBoundaryPoints(
44284                  Range.START_TO_START, nodeStartRange) == 1;
44285         
44286          
44287     },
44288     rangeCompareNode : function(range, node)
44289     {
44290         var nodeRange = node.ownerDocument.createRange();
44291         try {
44292             nodeRange.selectNode(node);
44293         } catch (e) {
44294             nodeRange.selectNodeContents(node);
44295         }
44296         
44297         
44298         range.collapse(true);
44299     
44300         nodeRange.collapse(true);
44301      
44302         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44303         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44304          
44305         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44306         
44307         var nodeIsBefore   =  ss == 1;
44308         var nodeIsAfter    = ee == -1;
44309         
44310         if (nodeIsBefore && nodeIsAfter) {
44311             return 0; // outer
44312         }
44313         if (!nodeIsBefore && nodeIsAfter) {
44314             return 1; //right trailed.
44315         }
44316         
44317         if (nodeIsBefore && !nodeIsAfter) {
44318             return 2;  // left trailed.
44319         }
44320         // fully contined.
44321         return 3;
44322     },
44323
44324     // private? - in a new class?
44325     cleanUpPaste :  function()
44326     {
44327         // cleans up the whole document..
44328         Roo.log('cleanuppaste');
44329         
44330         this.cleanUpChildren(this.doc.body);
44331         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44332         if (clean != this.doc.body.innerHTML) {
44333             this.doc.body.innerHTML = clean;
44334         }
44335         
44336     },
44337     
44338     cleanWordChars : function(input) {// change the chars to hex code
44339         var he = Roo.HtmlEditorCore;
44340         
44341         var output = input;
44342         Roo.each(he.swapCodes, function(sw) { 
44343             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44344             
44345             output = output.replace(swapper, sw[1]);
44346         });
44347         
44348         return output;
44349     },
44350     
44351     
44352     cleanUpChildren : function (n)
44353     {
44354         if (!n.childNodes.length) {
44355             return;
44356         }
44357         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44358            this.cleanUpChild(n.childNodes[i]);
44359         }
44360     },
44361     
44362     
44363         
44364     
44365     cleanUpChild : function (node)
44366     {
44367         var ed = this;
44368         //console.log(node);
44369         if (node.nodeName == "#text") {
44370             // clean up silly Windows -- stuff?
44371             return; 
44372         }
44373         if (node.nodeName == "#comment") {
44374             node.parentNode.removeChild(node);
44375             // clean up silly Windows -- stuff?
44376             return; 
44377         }
44378         var lcname = node.tagName.toLowerCase();
44379         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44380         // whitelist of tags..
44381         
44382         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44383             // remove node.
44384             node.parentNode.removeChild(node);
44385             return;
44386             
44387         }
44388         
44389         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44390         
44391         // spans with no attributes - just remove them..
44392         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44393             remove_keep_children = true;
44394         }
44395         
44396         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44397         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44398         
44399         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44400         //    remove_keep_children = true;
44401         //}
44402         
44403         if (remove_keep_children) {
44404             this.cleanUpChildren(node);
44405             // inserts everything just before this node...
44406             while (node.childNodes.length) {
44407                 var cn = node.childNodes[0];
44408                 node.removeChild(cn);
44409                 node.parentNode.insertBefore(cn, node);
44410             }
44411             node.parentNode.removeChild(node);
44412             return;
44413         }
44414         
44415         if (!node.attributes || !node.attributes.length) {
44416             
44417           
44418             
44419             
44420             this.cleanUpChildren(node);
44421             return;
44422         }
44423         
44424         function cleanAttr(n,v)
44425         {
44426             
44427             if (v.match(/^\./) || v.match(/^\//)) {
44428                 return;
44429             }
44430             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44431                 return;
44432             }
44433             if (v.match(/^#/)) {
44434                 return;
44435             }
44436 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44437             node.removeAttribute(n);
44438             
44439         }
44440         
44441         var cwhite = this.cwhite;
44442         var cblack = this.cblack;
44443             
44444         function cleanStyle(n,v)
44445         {
44446             if (v.match(/expression/)) { //XSS?? should we even bother..
44447                 node.removeAttribute(n);
44448                 return;
44449             }
44450             
44451             var parts = v.split(/;/);
44452             var clean = [];
44453             
44454             Roo.each(parts, function(p) {
44455                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44456                 if (!p.length) {
44457                     return true;
44458                 }
44459                 var l = p.split(':').shift().replace(/\s+/g,'');
44460                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44461                 
44462                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44463 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44464                     //node.removeAttribute(n);
44465                     return true;
44466                 }
44467                 //Roo.log()
44468                 // only allow 'c whitelisted system attributes'
44469                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44470 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44471                     //node.removeAttribute(n);
44472                     return true;
44473                 }
44474                 
44475                 
44476                  
44477                 
44478                 clean.push(p);
44479                 return true;
44480             });
44481             if (clean.length) { 
44482                 node.setAttribute(n, clean.join(';'));
44483             } else {
44484                 node.removeAttribute(n);
44485             }
44486             
44487         }
44488         
44489         
44490         for (var i = node.attributes.length-1; i > -1 ; i--) {
44491             var a = node.attributes[i];
44492             //console.log(a);
44493             
44494             if (a.name.toLowerCase().substr(0,2)=='on')  {
44495                 node.removeAttribute(a.name);
44496                 continue;
44497             }
44498             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44499                 node.removeAttribute(a.name);
44500                 continue;
44501             }
44502             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44503                 cleanAttr(a.name,a.value); // fixme..
44504                 continue;
44505             }
44506             if (a.name == 'style') {
44507                 cleanStyle(a.name,a.value);
44508                 continue;
44509             }
44510             /// clean up MS crap..
44511             // tecnically this should be a list of valid class'es..
44512             
44513             
44514             if (a.name == 'class') {
44515                 if (a.value.match(/^Mso/)) {
44516                     node.removeAttribute('class');
44517                 }
44518                 
44519                 if (a.value.match(/^body$/)) {
44520                     node.removeAttribute('class');
44521                 }
44522                 continue;
44523             }
44524             
44525             // style cleanup!?
44526             // class cleanup?
44527             
44528         }
44529         
44530         
44531         this.cleanUpChildren(node);
44532         
44533         
44534     },
44535     
44536     /**
44537      * Clean up MS wordisms...
44538      */
44539     cleanWord : function(node)
44540     {
44541         if (!node) {
44542             this.cleanWord(this.doc.body);
44543             return;
44544         }
44545         
44546         if(
44547                 node.nodeName == 'SPAN' &&
44548                 !node.hasAttributes() &&
44549                 node.childNodes.length == 1 &&
44550                 node.firstChild.nodeName == "#text"  
44551         ) {
44552             var textNode = node.firstChild;
44553             node.removeChild(textNode);
44554             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44555                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44556             }
44557             node.parentNode.insertBefore(textNode, node);
44558             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44559                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44560             }
44561             node.parentNode.removeChild(node);
44562         }
44563         
44564         if (node.nodeName == "#text") {
44565             // clean up silly Windows -- stuff?
44566             return; 
44567         }
44568         if (node.nodeName == "#comment") {
44569             node.parentNode.removeChild(node);
44570             // clean up silly Windows -- stuff?
44571             return; 
44572         }
44573         
44574         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44575             node.parentNode.removeChild(node);
44576             return;
44577         }
44578         //Roo.log(node.tagName);
44579         // remove - but keep children..
44580         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44581             //Roo.log('-- removed');
44582             while (node.childNodes.length) {
44583                 var cn = node.childNodes[0];
44584                 node.removeChild(cn);
44585                 node.parentNode.insertBefore(cn, node);
44586                 // move node to parent - and clean it..
44587                 this.cleanWord(cn);
44588             }
44589             node.parentNode.removeChild(node);
44590             /// no need to iterate chidlren = it's got none..
44591             //this.iterateChildren(node, this.cleanWord);
44592             return;
44593         }
44594         // clean styles
44595         if (node.className.length) {
44596             
44597             var cn = node.className.split(/\W+/);
44598             var cna = [];
44599             Roo.each(cn, function(cls) {
44600                 if (cls.match(/Mso[a-zA-Z]+/)) {
44601                     return;
44602                 }
44603                 cna.push(cls);
44604             });
44605             node.className = cna.length ? cna.join(' ') : '';
44606             if (!cna.length) {
44607                 node.removeAttribute("class");
44608             }
44609         }
44610         
44611         if (node.hasAttribute("lang")) {
44612             node.removeAttribute("lang");
44613         }
44614         
44615         if (node.hasAttribute("style")) {
44616             
44617             var styles = node.getAttribute("style").split(";");
44618             var nstyle = [];
44619             Roo.each(styles, function(s) {
44620                 if (!s.match(/:/)) {
44621                     return;
44622                 }
44623                 var kv = s.split(":");
44624                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44625                     return;
44626                 }
44627                 // what ever is left... we allow.
44628                 nstyle.push(s);
44629             });
44630             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44631             if (!nstyle.length) {
44632                 node.removeAttribute('style');
44633             }
44634         }
44635         this.iterateChildren(node, this.cleanWord);
44636         
44637         
44638         
44639     },
44640     /**
44641      * iterateChildren of a Node, calling fn each time, using this as the scole..
44642      * @param {DomNode} node node to iterate children of.
44643      * @param {Function} fn method of this class to call on each item.
44644      */
44645     iterateChildren : function(node, fn)
44646     {
44647         if (!node.childNodes.length) {
44648                 return;
44649         }
44650         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44651            fn.call(this, node.childNodes[i])
44652         }
44653     },
44654     
44655     
44656     /**
44657      * cleanTableWidths.
44658      *
44659      * Quite often pasting from word etc.. results in tables with column and widths.
44660      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44661      *
44662      */
44663     cleanTableWidths : function(node)
44664     {
44665          
44666          
44667         if (!node) {
44668             this.cleanTableWidths(this.doc.body);
44669             return;
44670         }
44671         
44672         // ignore list...
44673         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44674             return; 
44675         }
44676         Roo.log(node.tagName);
44677         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44678             this.iterateChildren(node, this.cleanTableWidths);
44679             return;
44680         }
44681         if (node.hasAttribute('width')) {
44682             node.removeAttribute('width');
44683         }
44684         
44685          
44686         if (node.hasAttribute("style")) {
44687             // pretty basic...
44688             
44689             var styles = node.getAttribute("style").split(";");
44690             var nstyle = [];
44691             Roo.each(styles, function(s) {
44692                 if (!s.match(/:/)) {
44693                     return;
44694                 }
44695                 var kv = s.split(":");
44696                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44697                     return;
44698                 }
44699                 // what ever is left... we allow.
44700                 nstyle.push(s);
44701             });
44702             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44703             if (!nstyle.length) {
44704                 node.removeAttribute('style');
44705             }
44706         }
44707         
44708         this.iterateChildren(node, this.cleanTableWidths);
44709         
44710         
44711     },
44712     
44713     
44714     
44715     
44716     domToHTML : function(currentElement, depth, nopadtext) {
44717         
44718         depth = depth || 0;
44719         nopadtext = nopadtext || false;
44720     
44721         if (!currentElement) {
44722             return this.domToHTML(this.doc.body);
44723         }
44724         
44725         //Roo.log(currentElement);
44726         var j;
44727         var allText = false;
44728         var nodeName = currentElement.nodeName;
44729         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44730         
44731         if  (nodeName == '#text') {
44732             
44733             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44734         }
44735         
44736         
44737         var ret = '';
44738         if (nodeName != 'BODY') {
44739              
44740             var i = 0;
44741             // Prints the node tagName, such as <A>, <IMG>, etc
44742             if (tagName) {
44743                 var attr = [];
44744                 for(i = 0; i < currentElement.attributes.length;i++) {
44745                     // quoting?
44746                     var aname = currentElement.attributes.item(i).name;
44747                     if (!currentElement.attributes.item(i).value.length) {
44748                         continue;
44749                     }
44750                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44751                 }
44752                 
44753                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44754             } 
44755             else {
44756                 
44757                 // eack
44758             }
44759         } else {
44760             tagName = false;
44761         }
44762         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44763             return ret;
44764         }
44765         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44766             nopadtext = true;
44767         }
44768         
44769         
44770         // Traverse the tree
44771         i = 0;
44772         var currentElementChild = currentElement.childNodes.item(i);
44773         var allText = true;
44774         var innerHTML  = '';
44775         lastnode = '';
44776         while (currentElementChild) {
44777             // Formatting code (indent the tree so it looks nice on the screen)
44778             var nopad = nopadtext;
44779             if (lastnode == 'SPAN') {
44780                 nopad  = true;
44781             }
44782             // text
44783             if  (currentElementChild.nodeName == '#text') {
44784                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44785                 toadd = nopadtext ? toadd : toadd.trim();
44786                 if (!nopad && toadd.length > 80) {
44787                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44788                 }
44789                 innerHTML  += toadd;
44790                 
44791                 i++;
44792                 currentElementChild = currentElement.childNodes.item(i);
44793                 lastNode = '';
44794                 continue;
44795             }
44796             allText = false;
44797             
44798             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44799                 
44800             // Recursively traverse the tree structure of the child node
44801             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44802             lastnode = currentElementChild.nodeName;
44803             i++;
44804             currentElementChild=currentElement.childNodes.item(i);
44805         }
44806         
44807         ret += innerHTML;
44808         
44809         if (!allText) {
44810                 // The remaining code is mostly for formatting the tree
44811             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44812         }
44813         
44814         
44815         if (tagName) {
44816             ret+= "</"+tagName+">";
44817         }
44818         return ret;
44819         
44820     },
44821         
44822     applyBlacklists : function()
44823     {
44824         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44825         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44826         
44827         this.white = [];
44828         this.black = [];
44829         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44830             if (b.indexOf(tag) > -1) {
44831                 return;
44832             }
44833             this.white.push(tag);
44834             
44835         }, this);
44836         
44837         Roo.each(w, function(tag) {
44838             if (b.indexOf(tag) > -1) {
44839                 return;
44840             }
44841             if (this.white.indexOf(tag) > -1) {
44842                 return;
44843             }
44844             this.white.push(tag);
44845             
44846         }, this);
44847         
44848         
44849         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44850             if (w.indexOf(tag) > -1) {
44851                 return;
44852             }
44853             this.black.push(tag);
44854             
44855         }, this);
44856         
44857         Roo.each(b, function(tag) {
44858             if (w.indexOf(tag) > -1) {
44859                 return;
44860             }
44861             if (this.black.indexOf(tag) > -1) {
44862                 return;
44863             }
44864             this.black.push(tag);
44865             
44866         }, this);
44867         
44868         
44869         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44870         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44871         
44872         this.cwhite = [];
44873         this.cblack = [];
44874         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44875             if (b.indexOf(tag) > -1) {
44876                 return;
44877             }
44878             this.cwhite.push(tag);
44879             
44880         }, this);
44881         
44882         Roo.each(w, function(tag) {
44883             if (b.indexOf(tag) > -1) {
44884                 return;
44885             }
44886             if (this.cwhite.indexOf(tag) > -1) {
44887                 return;
44888             }
44889             this.cwhite.push(tag);
44890             
44891         }, this);
44892         
44893         
44894         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44895             if (w.indexOf(tag) > -1) {
44896                 return;
44897             }
44898             this.cblack.push(tag);
44899             
44900         }, this);
44901         
44902         Roo.each(b, function(tag) {
44903             if (w.indexOf(tag) > -1) {
44904                 return;
44905             }
44906             if (this.cblack.indexOf(tag) > -1) {
44907                 return;
44908             }
44909             this.cblack.push(tag);
44910             
44911         }, this);
44912     },
44913     
44914     setStylesheets : function(stylesheets)
44915     {
44916         if(typeof(stylesheets) == 'string'){
44917             Roo.get(this.iframe.contentDocument.head).createChild({
44918                 tag : 'link',
44919                 rel : 'stylesheet',
44920                 type : 'text/css',
44921                 href : stylesheets
44922             });
44923             
44924             return;
44925         }
44926         var _this = this;
44927      
44928         Roo.each(stylesheets, function(s) {
44929             if(!s.length){
44930                 return;
44931             }
44932             
44933             Roo.get(_this.iframe.contentDocument.head).createChild({
44934                 tag : 'link',
44935                 rel : 'stylesheet',
44936                 type : 'text/css',
44937                 href : s
44938             });
44939         });
44940
44941         
44942     },
44943     
44944     removeStylesheets : function()
44945     {
44946         var _this = this;
44947         
44948         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44949             s.remove();
44950         });
44951     },
44952     
44953     setStyle : function(style)
44954     {
44955         Roo.get(this.iframe.contentDocument.head).createChild({
44956             tag : 'style',
44957             type : 'text/css',
44958             html : style
44959         });
44960
44961         return;
44962     }
44963     
44964     // hide stuff that is not compatible
44965     /**
44966      * @event blur
44967      * @hide
44968      */
44969     /**
44970      * @event change
44971      * @hide
44972      */
44973     /**
44974      * @event focus
44975      * @hide
44976      */
44977     /**
44978      * @event specialkey
44979      * @hide
44980      */
44981     /**
44982      * @cfg {String} fieldClass @hide
44983      */
44984     /**
44985      * @cfg {String} focusClass @hide
44986      */
44987     /**
44988      * @cfg {String} autoCreate @hide
44989      */
44990     /**
44991      * @cfg {String} inputType @hide
44992      */
44993     /**
44994      * @cfg {String} invalidClass @hide
44995      */
44996     /**
44997      * @cfg {String} invalidText @hide
44998      */
44999     /**
45000      * @cfg {String} msgFx @hide
45001      */
45002     /**
45003      * @cfg {String} validateOnBlur @hide
45004      */
45005 });
45006
45007 Roo.HtmlEditorCore.white = [
45008         'area', 'br', 'img', 'input', 'hr', 'wbr',
45009         
45010        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
45011        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
45012        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
45013        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
45014        'table',   'ul',         'xmp', 
45015        
45016        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
45017       'thead',   'tr', 
45018      
45019       'dir', 'menu', 'ol', 'ul', 'dl',
45020        
45021       'embed',  'object'
45022 ];
45023
45024
45025 Roo.HtmlEditorCore.black = [
45026     //    'embed',  'object', // enable - backend responsiblity to clean thiese
45027         'applet', // 
45028         'base',   'basefont', 'bgsound', 'blink',  'body', 
45029         'frame',  'frameset', 'head',    'html',   'ilayer', 
45030         'iframe', 'layer',  'link',     'meta',    'object',   
45031         'script', 'style' ,'title',  'xml' // clean later..
45032 ];
45033 Roo.HtmlEditorCore.clean = [
45034     'script', 'style', 'title', 'xml'
45035 ];
45036 Roo.HtmlEditorCore.remove = [
45037     'font'
45038 ];
45039 // attributes..
45040
45041 Roo.HtmlEditorCore.ablack = [
45042     'on'
45043 ];
45044     
45045 Roo.HtmlEditorCore.aclean = [ 
45046     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
45047 ];
45048
45049 // protocols..
45050 Roo.HtmlEditorCore.pwhite= [
45051         'http',  'https',  'mailto'
45052 ];
45053
45054 // white listed style attributes.
45055 Roo.HtmlEditorCore.cwhite= [
45056       //  'text-align', /// default is to allow most things..
45057       
45058          
45059 //        'font-size'//??
45060 ];
45061
45062 // black listed style attributes.
45063 Roo.HtmlEditorCore.cblack= [
45064       //  'font-size' -- this can be set by the project 
45065 ];
45066
45067
45068 Roo.HtmlEditorCore.swapCodes   =[ 
45069     [    8211, "--" ], 
45070     [    8212, "--" ], 
45071     [    8216,  "'" ],  
45072     [    8217, "'" ],  
45073     [    8220, '"' ],  
45074     [    8221, '"' ],  
45075     [    8226, "*" ],  
45076     [    8230, "..." ]
45077 ]; 
45078
45079     //<script type="text/javascript">
45080
45081 /*
45082  * Ext JS Library 1.1.1
45083  * Copyright(c) 2006-2007, Ext JS, LLC.
45084  * Licence LGPL
45085  * 
45086  */
45087  
45088  
45089 Roo.form.HtmlEditor = function(config){
45090     
45091     
45092     
45093     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45094     
45095     if (!this.toolbars) {
45096         this.toolbars = [];
45097     }
45098     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45099     
45100     
45101 };
45102
45103 /**
45104  * @class Roo.form.HtmlEditor
45105  * @extends Roo.form.Field
45106  * Provides a lightweight HTML Editor component.
45107  *
45108  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45109  * 
45110  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45111  * supported by this editor.</b><br/><br/>
45112  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45113  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45114  */
45115 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45116     /**
45117      * @cfg {Boolean} clearUp
45118      */
45119     clearUp : true,
45120       /**
45121      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45122      */
45123     toolbars : false,
45124    
45125      /**
45126      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45127      *                        Roo.resizable.
45128      */
45129     resizable : false,
45130      /**
45131      * @cfg {Number} height (in pixels)
45132      */   
45133     height: 300,
45134    /**
45135      * @cfg {Number} width (in pixels)
45136      */   
45137     width: 500,
45138     
45139     /**
45140      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45141      * 
45142      */
45143     stylesheets: false,
45144     
45145     
45146      /**
45147      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45148      * 
45149      */
45150     cblack: false,
45151     /**
45152      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45153      * 
45154      */
45155     cwhite: false,
45156     
45157      /**
45158      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45159      * 
45160      */
45161     black: false,
45162     /**
45163      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45164      * 
45165      */
45166     white: false,
45167     
45168     // id of frame..
45169     frameId: false,
45170     
45171     // private properties
45172     validationEvent : false,
45173     deferHeight: true,
45174     initialized : false,
45175     activated : false,
45176     
45177     onFocus : Roo.emptyFn,
45178     iframePad:3,
45179     hideMode:'offsets',
45180     
45181     actionMode : 'container', // defaults to hiding it...
45182     
45183     defaultAutoCreate : { // modified by initCompnoent..
45184         tag: "textarea",
45185         style:"width:500px;height:300px;",
45186         autocomplete: "new-password"
45187     },
45188
45189     // private
45190     initComponent : function(){
45191         this.addEvents({
45192             /**
45193              * @event initialize
45194              * Fires when the editor is fully initialized (including the iframe)
45195              * @param {HtmlEditor} this
45196              */
45197             initialize: true,
45198             /**
45199              * @event activate
45200              * Fires when the editor is first receives the focus. Any insertion must wait
45201              * until after this event.
45202              * @param {HtmlEditor} this
45203              */
45204             activate: true,
45205              /**
45206              * @event beforesync
45207              * Fires before the textarea is updated with content from the editor iframe. Return false
45208              * to cancel the sync.
45209              * @param {HtmlEditor} this
45210              * @param {String} html
45211              */
45212             beforesync: true,
45213              /**
45214              * @event beforepush
45215              * Fires before the iframe editor is updated with content from the textarea. Return false
45216              * to cancel the push.
45217              * @param {HtmlEditor} this
45218              * @param {String} html
45219              */
45220             beforepush: true,
45221              /**
45222              * @event sync
45223              * Fires when the textarea is updated with content from the editor iframe.
45224              * @param {HtmlEditor} this
45225              * @param {String} html
45226              */
45227             sync: true,
45228              /**
45229              * @event push
45230              * Fires when the iframe editor is updated with content from the textarea.
45231              * @param {HtmlEditor} this
45232              * @param {String} html
45233              */
45234             push: true,
45235              /**
45236              * @event editmodechange
45237              * Fires when the editor switches edit modes
45238              * @param {HtmlEditor} this
45239              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45240              */
45241             editmodechange: true,
45242             /**
45243              * @event editorevent
45244              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45245              * @param {HtmlEditor} this
45246              */
45247             editorevent: true,
45248             /**
45249              * @event firstfocus
45250              * Fires when on first focus - needed by toolbars..
45251              * @param {HtmlEditor} this
45252              */
45253             firstfocus: true,
45254             /**
45255              * @event autosave
45256              * Auto save the htmlEditor value as a file into Events
45257              * @param {HtmlEditor} this
45258              */
45259             autosave: true,
45260             /**
45261              * @event savedpreview
45262              * preview the saved version of htmlEditor
45263              * @param {HtmlEditor} this
45264              */
45265             savedpreview: true,
45266             
45267             /**
45268             * @event stylesheetsclick
45269             * Fires when press the Sytlesheets button
45270             * @param {Roo.HtmlEditorCore} this
45271             */
45272             stylesheetsclick: true
45273         });
45274         this.defaultAutoCreate =  {
45275             tag: "textarea",
45276             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45277             autocomplete: "new-password"
45278         };
45279     },
45280
45281     /**
45282      * Protected method that will not generally be called directly. It
45283      * is called when the editor creates its toolbar. Override this method if you need to
45284      * add custom toolbar buttons.
45285      * @param {HtmlEditor} editor
45286      */
45287     createToolbar : function(editor){
45288         Roo.log("create toolbars");
45289         if (!editor.toolbars || !editor.toolbars.length) {
45290             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45291         }
45292         
45293         for (var i =0 ; i < editor.toolbars.length;i++) {
45294             editor.toolbars[i] = Roo.factory(
45295                     typeof(editor.toolbars[i]) == 'string' ?
45296                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45297                 Roo.form.HtmlEditor);
45298             editor.toolbars[i].init(editor);
45299         }
45300          
45301         
45302     },
45303
45304      
45305     // private
45306     onRender : function(ct, position)
45307     {
45308         var _t = this;
45309         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45310         
45311         this.wrap = this.el.wrap({
45312             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45313         });
45314         
45315         this.editorcore.onRender(ct, position);
45316          
45317         if (this.resizable) {
45318             this.resizeEl = new Roo.Resizable(this.wrap, {
45319                 pinned : true,
45320                 wrap: true,
45321                 dynamic : true,
45322                 minHeight : this.height,
45323                 height: this.height,
45324                 handles : this.resizable,
45325                 width: this.width,
45326                 listeners : {
45327                     resize : function(r, w, h) {
45328                         _t.onResize(w,h); // -something
45329                     }
45330                 }
45331             });
45332             
45333         }
45334         this.createToolbar(this);
45335        
45336         
45337         if(!this.width){
45338             this.setSize(this.wrap.getSize());
45339         }
45340         if (this.resizeEl) {
45341             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45342             // should trigger onReize..
45343         }
45344         
45345         this.keyNav = new Roo.KeyNav(this.el, {
45346             
45347             "tab" : function(e){
45348                 e.preventDefault();
45349                 
45350                 var value = this.getValue();
45351                 
45352                 var start = this.el.dom.selectionStart;
45353                 var end = this.el.dom.selectionEnd;
45354                 
45355                 if(!e.shiftKey){
45356                     
45357                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45358                     this.el.dom.setSelectionRange(end + 1, end + 1);
45359                     return;
45360                 }
45361                 
45362                 var f = value.substring(0, start).split("\t");
45363                 
45364                 if(f.pop().length != 0){
45365                     return;
45366                 }
45367                 
45368                 this.setValue(f.join("\t") + value.substring(end));
45369                 this.el.dom.setSelectionRange(start - 1, start - 1);
45370                 
45371             },
45372             
45373             "home" : function(e){
45374                 e.preventDefault();
45375                 
45376                 var curr = this.el.dom.selectionStart;
45377                 var lines = this.getValue().split("\n");
45378                 
45379                 if(!lines.length){
45380                     return;
45381                 }
45382                 
45383                 if(e.ctrlKey){
45384                     this.el.dom.setSelectionRange(0, 0);
45385                     return;
45386                 }
45387                 
45388                 var pos = 0;
45389                 
45390                 for (var i = 0; i < lines.length;i++) {
45391                     pos += lines[i].length;
45392                     
45393                     if(i != 0){
45394                         pos += 1;
45395                     }
45396                     
45397                     if(pos < curr){
45398                         continue;
45399                     }
45400                     
45401                     pos -= lines[i].length;
45402                     
45403                     break;
45404                 }
45405                 
45406                 if(!e.shiftKey){
45407                     this.el.dom.setSelectionRange(pos, pos);
45408                     return;
45409                 }
45410                 
45411                 this.el.dom.selectionStart = pos;
45412                 this.el.dom.selectionEnd = curr;
45413             },
45414             
45415             "end" : function(e){
45416                 e.preventDefault();
45417                 
45418                 var curr = this.el.dom.selectionStart;
45419                 var lines = this.getValue().split("\n");
45420                 
45421                 if(!lines.length){
45422                     return;
45423                 }
45424                 
45425                 if(e.ctrlKey){
45426                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45427                     return;
45428                 }
45429                 
45430                 var pos = 0;
45431                 
45432                 for (var i = 0; i < lines.length;i++) {
45433                     
45434                     pos += lines[i].length;
45435                     
45436                     if(i != 0){
45437                         pos += 1;
45438                     }
45439                     
45440                     if(pos < curr){
45441                         continue;
45442                     }
45443                     
45444                     break;
45445                 }
45446                 
45447                 if(!e.shiftKey){
45448                     this.el.dom.setSelectionRange(pos, pos);
45449                     return;
45450                 }
45451                 
45452                 this.el.dom.selectionStart = curr;
45453                 this.el.dom.selectionEnd = pos;
45454             },
45455
45456             scope : this,
45457
45458             doRelay : function(foo, bar, hname){
45459                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45460             },
45461
45462             forceKeyDown: true
45463         });
45464         
45465 //        if(this.autosave && this.w){
45466 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45467 //        }
45468     },
45469
45470     // private
45471     onResize : function(w, h)
45472     {
45473         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45474         var ew = false;
45475         var eh = false;
45476         
45477         if(this.el ){
45478             if(typeof w == 'number'){
45479                 var aw = w - this.wrap.getFrameWidth('lr');
45480                 this.el.setWidth(this.adjustWidth('textarea', aw));
45481                 ew = aw;
45482             }
45483             if(typeof h == 'number'){
45484                 var tbh = 0;
45485                 for (var i =0; i < this.toolbars.length;i++) {
45486                     // fixme - ask toolbars for heights?
45487                     tbh += this.toolbars[i].tb.el.getHeight();
45488                     if (this.toolbars[i].footer) {
45489                         tbh += this.toolbars[i].footer.el.getHeight();
45490                     }
45491                 }
45492                 
45493                 
45494                 
45495                 
45496                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45497                 ah -= 5; // knock a few pixes off for look..
45498 //                Roo.log(ah);
45499                 this.el.setHeight(this.adjustWidth('textarea', ah));
45500                 var eh = ah;
45501             }
45502         }
45503         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45504         this.editorcore.onResize(ew,eh);
45505         
45506     },
45507
45508     /**
45509      * Toggles the editor between standard and source edit mode.
45510      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45511      */
45512     toggleSourceEdit : function(sourceEditMode)
45513     {
45514         this.editorcore.toggleSourceEdit(sourceEditMode);
45515         
45516         if(this.editorcore.sourceEditMode){
45517             Roo.log('editor - showing textarea');
45518             
45519 //            Roo.log('in');
45520 //            Roo.log(this.syncValue());
45521             this.editorcore.syncValue();
45522             this.el.removeClass('x-hidden');
45523             this.el.dom.removeAttribute('tabIndex');
45524             this.el.focus();
45525             
45526             for (var i = 0; i < this.toolbars.length; i++) {
45527                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45528                     this.toolbars[i].tb.hide();
45529                     this.toolbars[i].footer.hide();
45530                 }
45531             }
45532             
45533         }else{
45534             Roo.log('editor - hiding textarea');
45535 //            Roo.log('out')
45536 //            Roo.log(this.pushValue()); 
45537             this.editorcore.pushValue();
45538             
45539             this.el.addClass('x-hidden');
45540             this.el.dom.setAttribute('tabIndex', -1);
45541             
45542             for (var i = 0; i < this.toolbars.length; i++) {
45543                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45544                     this.toolbars[i].tb.show();
45545                     this.toolbars[i].footer.show();
45546                 }
45547             }
45548             
45549             //this.deferFocus();
45550         }
45551         
45552         this.setSize(this.wrap.getSize());
45553         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45554         
45555         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45556     },
45557  
45558     // private (for BoxComponent)
45559     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45560
45561     // private (for BoxComponent)
45562     getResizeEl : function(){
45563         return this.wrap;
45564     },
45565
45566     // private (for BoxComponent)
45567     getPositionEl : function(){
45568         return this.wrap;
45569     },
45570
45571     // private
45572     initEvents : function(){
45573         this.originalValue = this.getValue();
45574     },
45575
45576     /**
45577      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45578      * @method
45579      */
45580     markInvalid : Roo.emptyFn,
45581     /**
45582      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45583      * @method
45584      */
45585     clearInvalid : Roo.emptyFn,
45586
45587     setValue : function(v){
45588         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45589         this.editorcore.pushValue();
45590     },
45591
45592      
45593     // private
45594     deferFocus : function(){
45595         this.focus.defer(10, this);
45596     },
45597
45598     // doc'ed in Field
45599     focus : function(){
45600         this.editorcore.focus();
45601         
45602     },
45603       
45604
45605     // private
45606     onDestroy : function(){
45607         
45608         
45609         
45610         if(this.rendered){
45611             
45612             for (var i =0; i < this.toolbars.length;i++) {
45613                 // fixme - ask toolbars for heights?
45614                 this.toolbars[i].onDestroy();
45615             }
45616             
45617             this.wrap.dom.innerHTML = '';
45618             this.wrap.remove();
45619         }
45620     },
45621
45622     // private
45623     onFirstFocus : function(){
45624         //Roo.log("onFirstFocus");
45625         this.editorcore.onFirstFocus();
45626          for (var i =0; i < this.toolbars.length;i++) {
45627             this.toolbars[i].onFirstFocus();
45628         }
45629         
45630     },
45631     
45632     // private
45633     syncValue : function()
45634     {
45635         this.editorcore.syncValue();
45636     },
45637     
45638     pushValue : function()
45639     {
45640         this.editorcore.pushValue();
45641     },
45642     
45643     setStylesheets : function(stylesheets)
45644     {
45645         this.editorcore.setStylesheets(stylesheets);
45646     },
45647     
45648     removeStylesheets : function()
45649     {
45650         this.editorcore.removeStylesheets();
45651     }
45652      
45653     
45654     // hide stuff that is not compatible
45655     /**
45656      * @event blur
45657      * @hide
45658      */
45659     /**
45660      * @event change
45661      * @hide
45662      */
45663     /**
45664      * @event focus
45665      * @hide
45666      */
45667     /**
45668      * @event specialkey
45669      * @hide
45670      */
45671     /**
45672      * @cfg {String} fieldClass @hide
45673      */
45674     /**
45675      * @cfg {String} focusClass @hide
45676      */
45677     /**
45678      * @cfg {String} autoCreate @hide
45679      */
45680     /**
45681      * @cfg {String} inputType @hide
45682      */
45683     /**
45684      * @cfg {String} invalidClass @hide
45685      */
45686     /**
45687      * @cfg {String} invalidText @hide
45688      */
45689     /**
45690      * @cfg {String} msgFx @hide
45691      */
45692     /**
45693      * @cfg {String} validateOnBlur @hide
45694      */
45695 });
45696  
45697     // <script type="text/javascript">
45698 /*
45699  * Based on
45700  * Ext JS Library 1.1.1
45701  * Copyright(c) 2006-2007, Ext JS, LLC.
45702  *  
45703  
45704  */
45705
45706 /**
45707  * @class Roo.form.HtmlEditorToolbar1
45708  * Basic Toolbar
45709  * 
45710  * Usage:
45711  *
45712  new Roo.form.HtmlEditor({
45713     ....
45714     toolbars : [
45715         new Roo.form.HtmlEditorToolbar1({
45716             disable : { fonts: 1 , format: 1, ..., ... , ...],
45717             btns : [ .... ]
45718         })
45719     }
45720      
45721  * 
45722  * @cfg {Object} disable List of elements to disable..
45723  * @cfg {Array} btns List of additional buttons.
45724  * 
45725  * 
45726  * NEEDS Extra CSS? 
45727  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45728  */
45729  
45730 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45731 {
45732     
45733     Roo.apply(this, config);
45734     
45735     // default disabled, based on 'good practice'..
45736     this.disable = this.disable || {};
45737     Roo.applyIf(this.disable, {
45738         fontSize : true,
45739         colors : true,
45740         specialElements : true
45741     });
45742     
45743     
45744     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45745     // dont call parent... till later.
45746 }
45747
45748 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45749     
45750     tb: false,
45751     
45752     rendered: false,
45753     
45754     editor : false,
45755     editorcore : false,
45756     /**
45757      * @cfg {Object} disable  List of toolbar elements to disable
45758          
45759      */
45760     disable : false,
45761     
45762     
45763      /**
45764      * @cfg {String} createLinkText The default text for the create link prompt
45765      */
45766     createLinkText : 'Please enter the URL for the link:',
45767     /**
45768      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45769      */
45770     defaultLinkValue : 'http:/'+'/',
45771    
45772     
45773       /**
45774      * @cfg {Array} fontFamilies An array of available font families
45775      */
45776     fontFamilies : [
45777         'Arial',
45778         'Courier New',
45779         'Tahoma',
45780         'Times New Roman',
45781         'Verdana'
45782     ],
45783     
45784     specialChars : [
45785            "&#169;",
45786           "&#174;",     
45787           "&#8482;",    
45788           "&#163;" ,    
45789          // "&#8212;",    
45790           "&#8230;",    
45791           "&#247;" ,    
45792         //  "&#225;" ,     ?? a acute?
45793            "&#8364;"    , //Euro
45794        //   "&#8220;"    ,
45795         //  "&#8221;"    ,
45796         //  "&#8226;"    ,
45797           "&#176;"  //   , // degrees
45798
45799          // "&#233;"     , // e ecute
45800          // "&#250;"     , // u ecute?
45801     ],
45802     
45803     specialElements : [
45804         {
45805             text: "Insert Table",
45806             xtype: 'MenuItem',
45807             xns : Roo.Menu,
45808             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45809                 
45810         },
45811         {    
45812             text: "Insert Image",
45813             xtype: 'MenuItem',
45814             xns : Roo.Menu,
45815             ihtml : '<img src="about:blank"/>'
45816             
45817         }
45818         
45819          
45820     ],
45821     
45822     
45823     inputElements : [ 
45824             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45825             "input:submit", "input:button", "select", "textarea", "label" ],
45826     formats : [
45827         ["p"] ,  
45828         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45829         ["pre"],[ "code"], 
45830         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45831         ['div'],['span'],
45832         ['sup'],['sub']
45833     ],
45834     
45835     cleanStyles : [
45836         "font-size"
45837     ],
45838      /**
45839      * @cfg {String} defaultFont default font to use.
45840      */
45841     defaultFont: 'tahoma',
45842    
45843     fontSelect : false,
45844     
45845     
45846     formatCombo : false,
45847     
45848     init : function(editor)
45849     {
45850         this.editor = editor;
45851         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45852         var editorcore = this.editorcore;
45853         
45854         var _t = this;
45855         
45856         var fid = editorcore.frameId;
45857         var etb = this;
45858         function btn(id, toggle, handler){
45859             var xid = fid + '-'+ id ;
45860             return {
45861                 id : xid,
45862                 cmd : id,
45863                 cls : 'x-btn-icon x-edit-'+id,
45864                 enableToggle:toggle !== false,
45865                 scope: _t, // was editor...
45866                 handler:handler||_t.relayBtnCmd,
45867                 clickEvent:'mousedown',
45868                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45869                 tabIndex:-1
45870             };
45871         }
45872         
45873         
45874         
45875         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45876         this.tb = tb;
45877          // stop form submits
45878         tb.el.on('click', function(e){
45879             e.preventDefault(); // what does this do?
45880         });
45881
45882         if(!this.disable.font) { // && !Roo.isSafari){
45883             /* why no safari for fonts 
45884             editor.fontSelect = tb.el.createChild({
45885                 tag:'select',
45886                 tabIndex: -1,
45887                 cls:'x-font-select',
45888                 html: this.createFontOptions()
45889             });
45890             
45891             editor.fontSelect.on('change', function(){
45892                 var font = editor.fontSelect.dom.value;
45893                 editor.relayCmd('fontname', font);
45894                 editor.deferFocus();
45895             }, editor);
45896             
45897             tb.add(
45898                 editor.fontSelect.dom,
45899                 '-'
45900             );
45901             */
45902             
45903         };
45904         if(!this.disable.formats){
45905             this.formatCombo = new Roo.form.ComboBox({
45906                 store: new Roo.data.SimpleStore({
45907                     id : 'tag',
45908                     fields: ['tag'],
45909                     data : this.formats // from states.js
45910                 }),
45911                 blockFocus : true,
45912                 name : '',
45913                 //autoCreate : {tag: "div",  size: "20"},
45914                 displayField:'tag',
45915                 typeAhead: false,
45916                 mode: 'local',
45917                 editable : false,
45918                 triggerAction: 'all',
45919                 emptyText:'Add tag',
45920                 selectOnFocus:true,
45921                 width:135,
45922                 listeners : {
45923                     'select': function(c, r, i) {
45924                         editorcore.insertTag(r.get('tag'));
45925                         editor.focus();
45926                     }
45927                 }
45928
45929             });
45930             tb.addField(this.formatCombo);
45931             
45932         }
45933         
45934         if(!this.disable.format){
45935             tb.add(
45936                 btn('bold'),
45937                 btn('italic'),
45938                 btn('underline'),
45939                 btn('strikethrough')
45940             );
45941         };
45942         if(!this.disable.fontSize){
45943             tb.add(
45944                 '-',
45945                 
45946                 
45947                 btn('increasefontsize', false, editorcore.adjustFont),
45948                 btn('decreasefontsize', false, editorcore.adjustFont)
45949             );
45950         };
45951         
45952         
45953         if(!this.disable.colors){
45954             tb.add(
45955                 '-', {
45956                     id:editorcore.frameId +'-forecolor',
45957                     cls:'x-btn-icon x-edit-forecolor',
45958                     clickEvent:'mousedown',
45959                     tooltip: this.buttonTips['forecolor'] || undefined,
45960                     tabIndex:-1,
45961                     menu : new Roo.menu.ColorMenu({
45962                         allowReselect: true,
45963                         focus: Roo.emptyFn,
45964                         value:'000000',
45965                         plain:true,
45966                         selectHandler: function(cp, color){
45967                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45968                             editor.deferFocus();
45969                         },
45970                         scope: editorcore,
45971                         clickEvent:'mousedown'
45972                     })
45973                 }, {
45974                     id:editorcore.frameId +'backcolor',
45975                     cls:'x-btn-icon x-edit-backcolor',
45976                     clickEvent:'mousedown',
45977                     tooltip: this.buttonTips['backcolor'] || undefined,
45978                     tabIndex:-1,
45979                     menu : new Roo.menu.ColorMenu({
45980                         focus: Roo.emptyFn,
45981                         value:'FFFFFF',
45982                         plain:true,
45983                         allowReselect: true,
45984                         selectHandler: function(cp, color){
45985                             if(Roo.isGecko){
45986                                 editorcore.execCmd('useCSS', false);
45987                                 editorcore.execCmd('hilitecolor', color);
45988                                 editorcore.execCmd('useCSS', true);
45989                                 editor.deferFocus();
45990                             }else{
45991                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45992                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45993                                 editor.deferFocus();
45994                             }
45995                         },
45996                         scope:editorcore,
45997                         clickEvent:'mousedown'
45998                     })
45999                 }
46000             );
46001         };
46002         // now add all the items...
46003         
46004
46005         if(!this.disable.alignments){
46006             tb.add(
46007                 '-',
46008                 btn('justifyleft'),
46009                 btn('justifycenter'),
46010                 btn('justifyright')
46011             );
46012         };
46013
46014         //if(!Roo.isSafari){
46015             if(!this.disable.links){
46016                 tb.add(
46017                     '-',
46018                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
46019                 );
46020             };
46021
46022             if(!this.disable.lists){
46023                 tb.add(
46024                     '-',
46025                     btn('insertorderedlist'),
46026                     btn('insertunorderedlist')
46027                 );
46028             }
46029             if(!this.disable.sourceEdit){
46030                 tb.add(
46031                     '-',
46032                     btn('sourceedit', true, function(btn){
46033                         this.toggleSourceEdit(btn.pressed);
46034                     })
46035                 );
46036             }
46037         //}
46038         
46039         var smenu = { };
46040         // special menu.. - needs to be tidied up..
46041         if (!this.disable.special) {
46042             smenu = {
46043                 text: "&#169;",
46044                 cls: 'x-edit-none',
46045                 
46046                 menu : {
46047                     items : []
46048                 }
46049             };
46050             for (var i =0; i < this.specialChars.length; i++) {
46051                 smenu.menu.items.push({
46052                     
46053                     html: this.specialChars[i],
46054                     handler: function(a,b) {
46055                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
46056                         //editor.insertAtCursor(a.html);
46057                         
46058                     },
46059                     tabIndex:-1
46060                 });
46061             }
46062             
46063             
46064             tb.add(smenu);
46065             
46066             
46067         }
46068         
46069         var cmenu = { };
46070         if (!this.disable.cleanStyles) {
46071             cmenu = {
46072                 cls: 'x-btn-icon x-btn-clear',
46073                 
46074                 menu : {
46075                     items : []
46076                 }
46077             };
46078             for (var i =0; i < this.cleanStyles.length; i++) {
46079                 cmenu.menu.items.push({
46080                     actiontype : this.cleanStyles[i],
46081                     html: 'Remove ' + this.cleanStyles[i],
46082                     handler: function(a,b) {
46083 //                        Roo.log(a);
46084 //                        Roo.log(b);
46085                         var c = Roo.get(editorcore.doc.body);
46086                         c.select('[style]').each(function(s) {
46087                             s.dom.style.removeProperty(a.actiontype);
46088                         });
46089                         editorcore.syncValue();
46090                     },
46091                     tabIndex:-1
46092                 });
46093             }
46094              cmenu.menu.items.push({
46095                 actiontype : 'tablewidths',
46096                 html: 'Remove Table Widths',
46097                 handler: function(a,b) {
46098                     editorcore.cleanTableWidths();
46099                     editorcore.syncValue();
46100                 },
46101                 tabIndex:-1
46102             });
46103             cmenu.menu.items.push({
46104                 actiontype : 'word',
46105                 html: 'Remove MS Word Formating',
46106                 handler: function(a,b) {
46107                     editorcore.cleanWord();
46108                     editorcore.syncValue();
46109                 },
46110                 tabIndex:-1
46111             });
46112             
46113             cmenu.menu.items.push({
46114                 actiontype : 'all',
46115                 html: 'Remove All Styles',
46116                 handler: function(a,b) {
46117                     
46118                     var c = Roo.get(editorcore.doc.body);
46119                     c.select('[style]').each(function(s) {
46120                         s.dom.removeAttribute('style');
46121                     });
46122                     editorcore.syncValue();
46123                 },
46124                 tabIndex:-1
46125             });
46126             
46127             cmenu.menu.items.push({
46128                 actiontype : 'all',
46129                 html: 'Remove All CSS Classes',
46130                 handler: function(a,b) {
46131                     
46132                     var c = Roo.get(editorcore.doc.body);
46133                     c.select('[class]').each(function(s) {
46134                         s.dom.removeAttribute('class');
46135                     });
46136                     editorcore.cleanWord();
46137                     editorcore.syncValue();
46138                 },
46139                 tabIndex:-1
46140             });
46141             
46142              cmenu.menu.items.push({
46143                 actiontype : 'tidy',
46144                 html: 'Tidy HTML Source',
46145                 handler: function(a,b) {
46146                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46147                     editorcore.syncValue();
46148                 },
46149                 tabIndex:-1
46150             });
46151             
46152             
46153             tb.add(cmenu);
46154         }
46155          
46156         if (!this.disable.specialElements) {
46157             var semenu = {
46158                 text: "Other;",
46159                 cls: 'x-edit-none',
46160                 menu : {
46161                     items : []
46162                 }
46163             };
46164             for (var i =0; i < this.specialElements.length; i++) {
46165                 semenu.menu.items.push(
46166                     Roo.apply({ 
46167                         handler: function(a,b) {
46168                             editor.insertAtCursor(this.ihtml);
46169                         }
46170                     }, this.specialElements[i])
46171                 );
46172                     
46173             }
46174             
46175             tb.add(semenu);
46176             
46177             
46178         }
46179          
46180         
46181         if (this.btns) {
46182             for(var i =0; i< this.btns.length;i++) {
46183                 var b = Roo.factory(this.btns[i],Roo.form);
46184                 b.cls =  'x-edit-none';
46185                 
46186                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46187                     b.cls += ' x-init-enable';
46188                 }
46189                 
46190                 b.scope = editorcore;
46191                 tb.add(b);
46192             }
46193         
46194         }
46195         
46196         
46197         
46198         // disable everything...
46199         
46200         this.tb.items.each(function(item){
46201             
46202            if(
46203                 item.id != editorcore.frameId+ '-sourceedit' && 
46204                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46205             ){
46206                 
46207                 item.disable();
46208             }
46209         });
46210         this.rendered = true;
46211         
46212         // the all the btns;
46213         editor.on('editorevent', this.updateToolbar, this);
46214         // other toolbars need to implement this..
46215         //editor.on('editmodechange', this.updateToolbar, this);
46216     },
46217     
46218     
46219     relayBtnCmd : function(btn) {
46220         this.editorcore.relayCmd(btn.cmd);
46221     },
46222     // private used internally
46223     createLink : function(){
46224         Roo.log("create link?");
46225         var url = prompt(this.createLinkText, this.defaultLinkValue);
46226         if(url && url != 'http:/'+'/'){
46227             this.editorcore.relayCmd('createlink', url);
46228         }
46229     },
46230
46231     
46232     /**
46233      * Protected method that will not generally be called directly. It triggers
46234      * a toolbar update by reading the markup state of the current selection in the editor.
46235      */
46236     updateToolbar: function(){
46237
46238         if(!this.editorcore.activated){
46239             this.editor.onFirstFocus();
46240             return;
46241         }
46242
46243         var btns = this.tb.items.map, 
46244             doc = this.editorcore.doc,
46245             frameId = this.editorcore.frameId;
46246
46247         if(!this.disable.font && !Roo.isSafari){
46248             /*
46249             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46250             if(name != this.fontSelect.dom.value){
46251                 this.fontSelect.dom.value = name;
46252             }
46253             */
46254         }
46255         if(!this.disable.format){
46256             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46257             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46258             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46259             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46260         }
46261         if(!this.disable.alignments){
46262             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46263             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46264             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46265         }
46266         if(!Roo.isSafari && !this.disable.lists){
46267             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46268             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46269         }
46270         
46271         var ans = this.editorcore.getAllAncestors();
46272         if (this.formatCombo) {
46273             
46274             
46275             var store = this.formatCombo.store;
46276             this.formatCombo.setValue("");
46277             for (var i =0; i < ans.length;i++) {
46278                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46279                     // select it..
46280                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46281                     break;
46282                 }
46283             }
46284         }
46285         
46286         
46287         
46288         // hides menus... - so this cant be on a menu...
46289         Roo.menu.MenuMgr.hideAll();
46290
46291         //this.editorsyncValue();
46292     },
46293    
46294     
46295     createFontOptions : function(){
46296         var buf = [], fs = this.fontFamilies, ff, lc;
46297         
46298         
46299         
46300         for(var i = 0, len = fs.length; i< len; i++){
46301             ff = fs[i];
46302             lc = ff.toLowerCase();
46303             buf.push(
46304                 '<option value="',lc,'" style="font-family:',ff,';"',
46305                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46306                     ff,
46307                 '</option>'
46308             );
46309         }
46310         return buf.join('');
46311     },
46312     
46313     toggleSourceEdit : function(sourceEditMode){
46314         
46315         Roo.log("toolbar toogle");
46316         if(sourceEditMode === undefined){
46317             sourceEditMode = !this.sourceEditMode;
46318         }
46319         this.sourceEditMode = sourceEditMode === true;
46320         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46321         // just toggle the button?
46322         if(btn.pressed !== this.sourceEditMode){
46323             btn.toggle(this.sourceEditMode);
46324             return;
46325         }
46326         
46327         if(sourceEditMode){
46328             Roo.log("disabling buttons");
46329             this.tb.items.each(function(item){
46330                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46331                     item.disable();
46332                 }
46333             });
46334           
46335         }else{
46336             Roo.log("enabling buttons");
46337             if(this.editorcore.initialized){
46338                 this.tb.items.each(function(item){
46339                     item.enable();
46340                 });
46341             }
46342             
46343         }
46344         Roo.log("calling toggole on editor");
46345         // tell the editor that it's been pressed..
46346         this.editor.toggleSourceEdit(sourceEditMode);
46347        
46348     },
46349      /**
46350      * Object collection of toolbar tooltips for the buttons in the editor. The key
46351      * is the command id associated with that button and the value is a valid QuickTips object.
46352      * For example:
46353 <pre><code>
46354 {
46355     bold : {
46356         title: 'Bold (Ctrl+B)',
46357         text: 'Make the selected text bold.',
46358         cls: 'x-html-editor-tip'
46359     },
46360     italic : {
46361         title: 'Italic (Ctrl+I)',
46362         text: 'Make the selected text italic.',
46363         cls: 'x-html-editor-tip'
46364     },
46365     ...
46366 </code></pre>
46367     * @type Object
46368      */
46369     buttonTips : {
46370         bold : {
46371             title: 'Bold (Ctrl+B)',
46372             text: 'Make the selected text bold.',
46373             cls: 'x-html-editor-tip'
46374         },
46375         italic : {
46376             title: 'Italic (Ctrl+I)',
46377             text: 'Make the selected text italic.',
46378             cls: 'x-html-editor-tip'
46379         },
46380         underline : {
46381             title: 'Underline (Ctrl+U)',
46382             text: 'Underline the selected text.',
46383             cls: 'x-html-editor-tip'
46384         },
46385         strikethrough : {
46386             title: 'Strikethrough',
46387             text: 'Strikethrough the selected text.',
46388             cls: 'x-html-editor-tip'
46389         },
46390         increasefontsize : {
46391             title: 'Grow Text',
46392             text: 'Increase the font size.',
46393             cls: 'x-html-editor-tip'
46394         },
46395         decreasefontsize : {
46396             title: 'Shrink Text',
46397             text: 'Decrease the font size.',
46398             cls: 'x-html-editor-tip'
46399         },
46400         backcolor : {
46401             title: 'Text Highlight Color',
46402             text: 'Change the background color of the selected text.',
46403             cls: 'x-html-editor-tip'
46404         },
46405         forecolor : {
46406             title: 'Font Color',
46407             text: 'Change the color of the selected text.',
46408             cls: 'x-html-editor-tip'
46409         },
46410         justifyleft : {
46411             title: 'Align Text Left',
46412             text: 'Align text to the left.',
46413             cls: 'x-html-editor-tip'
46414         },
46415         justifycenter : {
46416             title: 'Center Text',
46417             text: 'Center text in the editor.',
46418             cls: 'x-html-editor-tip'
46419         },
46420         justifyright : {
46421             title: 'Align Text Right',
46422             text: 'Align text to the right.',
46423             cls: 'x-html-editor-tip'
46424         },
46425         insertunorderedlist : {
46426             title: 'Bullet List',
46427             text: 'Start a bulleted list.',
46428             cls: 'x-html-editor-tip'
46429         },
46430         insertorderedlist : {
46431             title: 'Numbered List',
46432             text: 'Start a numbered list.',
46433             cls: 'x-html-editor-tip'
46434         },
46435         createlink : {
46436             title: 'Hyperlink',
46437             text: 'Make the selected text a hyperlink.',
46438             cls: 'x-html-editor-tip'
46439         },
46440         sourceedit : {
46441             title: 'Source Edit',
46442             text: 'Switch to source editing mode.',
46443             cls: 'x-html-editor-tip'
46444         }
46445     },
46446     // private
46447     onDestroy : function(){
46448         if(this.rendered){
46449             
46450             this.tb.items.each(function(item){
46451                 if(item.menu){
46452                     item.menu.removeAll();
46453                     if(item.menu.el){
46454                         item.menu.el.destroy();
46455                     }
46456                 }
46457                 item.destroy();
46458             });
46459              
46460         }
46461     },
46462     onFirstFocus: function() {
46463         this.tb.items.each(function(item){
46464            item.enable();
46465         });
46466     }
46467 });
46468
46469
46470
46471
46472 // <script type="text/javascript">
46473 /*
46474  * Based on
46475  * Ext JS Library 1.1.1
46476  * Copyright(c) 2006-2007, Ext JS, LLC.
46477  *  
46478  
46479  */
46480
46481  
46482 /**
46483  * @class Roo.form.HtmlEditor.ToolbarContext
46484  * Context Toolbar
46485  * 
46486  * Usage:
46487  *
46488  new Roo.form.HtmlEditor({
46489     ....
46490     toolbars : [
46491         { xtype: 'ToolbarStandard', styles : {} }
46492         { xtype: 'ToolbarContext', disable : {} }
46493     ]
46494 })
46495
46496      
46497  * 
46498  * @config : {Object} disable List of elements to disable.. (not done yet.)
46499  * @config : {Object} styles  Map of styles available.
46500  * 
46501  */
46502
46503 Roo.form.HtmlEditor.ToolbarContext = function(config)
46504 {
46505     
46506     Roo.apply(this, config);
46507     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46508     // dont call parent... till later.
46509     this.styles = this.styles || {};
46510 }
46511
46512  
46513
46514 Roo.form.HtmlEditor.ToolbarContext.types = {
46515     'IMG' : {
46516         width : {
46517             title: "Width",
46518             width: 40
46519         },
46520         height:  {
46521             title: "Height",
46522             width: 40
46523         },
46524         align: {
46525             title: "Align",
46526             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46527             width : 80
46528             
46529         },
46530         border: {
46531             title: "Border",
46532             width: 40
46533         },
46534         alt: {
46535             title: "Alt",
46536             width: 120
46537         },
46538         src : {
46539             title: "Src",
46540             width: 220
46541         }
46542         
46543     },
46544     'A' : {
46545         name : {
46546             title: "Name",
46547             width: 50
46548         },
46549         target:  {
46550             title: "Target",
46551             width: 120
46552         },
46553         href:  {
46554             title: "Href",
46555             width: 220
46556         } // border?
46557         
46558     },
46559     'TABLE' : {
46560         rows : {
46561             title: "Rows",
46562             width: 20
46563         },
46564         cols : {
46565             title: "Cols",
46566             width: 20
46567         },
46568         width : {
46569             title: "Width",
46570             width: 40
46571         },
46572         height : {
46573             title: "Height",
46574             width: 40
46575         },
46576         border : {
46577             title: "Border",
46578             width: 20
46579         }
46580     },
46581     'TD' : {
46582         width : {
46583             title: "Width",
46584             width: 40
46585         },
46586         height : {
46587             title: "Height",
46588             width: 40
46589         },   
46590         align: {
46591             title: "Align",
46592             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46593             width: 80
46594         },
46595         valign: {
46596             title: "Valign",
46597             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46598             width: 80
46599         },
46600         colspan: {
46601             title: "Colspan",
46602             width: 20
46603             
46604         },
46605          'font-family'  : {
46606             title : "Font",
46607             style : 'fontFamily',
46608             displayField: 'display',
46609             optname : 'font-family',
46610             width: 140
46611         }
46612     },
46613     'INPUT' : {
46614         name : {
46615             title: "name",
46616             width: 120
46617         },
46618         value : {
46619             title: "Value",
46620             width: 120
46621         },
46622         width : {
46623             title: "Width",
46624             width: 40
46625         }
46626     },
46627     'LABEL' : {
46628         'for' : {
46629             title: "For",
46630             width: 120
46631         }
46632     },
46633     'TEXTAREA' : {
46634           name : {
46635             title: "name",
46636             width: 120
46637         },
46638         rows : {
46639             title: "Rows",
46640             width: 20
46641         },
46642         cols : {
46643             title: "Cols",
46644             width: 20
46645         }
46646     },
46647     'SELECT' : {
46648         name : {
46649             title: "name",
46650             width: 120
46651         },
46652         selectoptions : {
46653             title: "Options",
46654             width: 200
46655         }
46656     },
46657     
46658     // should we really allow this??
46659     // should this just be 
46660     'BODY' : {
46661         title : {
46662             title: "Title",
46663             width: 200,
46664             disabled : true
46665         }
46666     },
46667     'SPAN' : {
46668         'font-family'  : {
46669             title : "Font",
46670             style : 'fontFamily',
46671             displayField: 'display',
46672             optname : 'font-family',
46673             width: 140
46674         }
46675     },
46676     'DIV' : {
46677         'font-family'  : {
46678             title : "Font",
46679             style : 'fontFamily',
46680             displayField: 'display',
46681             optname : 'font-family',
46682             width: 140
46683         }
46684     },
46685      'P' : {
46686         'font-family'  : {
46687             title : "Font",
46688             style : 'fontFamily',
46689             displayField: 'display',
46690             optname : 'font-family',
46691             width: 140
46692         }
46693     },
46694     
46695     '*' : {
46696         // empty..
46697     }
46698
46699 };
46700
46701 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46702 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46703
46704 Roo.form.HtmlEditor.ToolbarContext.options = {
46705         'font-family'  : [ 
46706                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46707                 [ 'Courier New', 'Courier New'],
46708                 [ 'Tahoma', 'Tahoma'],
46709                 [ 'Times New Roman,serif', 'Times'],
46710                 [ 'Verdana','Verdana' ]
46711         ]
46712 };
46713
46714 // fixme - these need to be configurable..
46715  
46716
46717 //Roo.form.HtmlEditor.ToolbarContext.types
46718
46719
46720 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46721     
46722     tb: false,
46723     
46724     rendered: false,
46725     
46726     editor : false,
46727     editorcore : false,
46728     /**
46729      * @cfg {Object} disable  List of toolbar elements to disable
46730          
46731      */
46732     disable : false,
46733     /**
46734      * @cfg {Object} styles List of styles 
46735      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46736      *
46737      * These must be defined in the page, so they get rendered correctly..
46738      * .headline { }
46739      * TD.underline { }
46740      * 
46741      */
46742     styles : false,
46743     
46744     options: false,
46745     
46746     toolbars : false,
46747     
46748     init : function(editor)
46749     {
46750         this.editor = editor;
46751         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46752         var editorcore = this.editorcore;
46753         
46754         var fid = editorcore.frameId;
46755         var etb = this;
46756         function btn(id, toggle, handler){
46757             var xid = fid + '-'+ id ;
46758             return {
46759                 id : xid,
46760                 cmd : id,
46761                 cls : 'x-btn-icon x-edit-'+id,
46762                 enableToggle:toggle !== false,
46763                 scope: editorcore, // was editor...
46764                 handler:handler||editorcore.relayBtnCmd,
46765                 clickEvent:'mousedown',
46766                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46767                 tabIndex:-1
46768             };
46769         }
46770         // create a new element.
46771         var wdiv = editor.wrap.createChild({
46772                 tag: 'div'
46773             }, editor.wrap.dom.firstChild.nextSibling, true);
46774         
46775         // can we do this more than once??
46776         
46777          // stop form submits
46778       
46779  
46780         // disable everything...
46781         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46782         this.toolbars = {};
46783            
46784         for (var i in  ty) {
46785           
46786             this.toolbars[i] = this.buildToolbar(ty[i],i);
46787         }
46788         this.tb = this.toolbars.BODY;
46789         this.tb.el.show();
46790         this.buildFooter();
46791         this.footer.show();
46792         editor.on('hide', function( ) { this.footer.hide() }, this);
46793         editor.on('show', function( ) { this.footer.show() }, this);
46794         
46795          
46796         this.rendered = true;
46797         
46798         // the all the btns;
46799         editor.on('editorevent', this.updateToolbar, this);
46800         // other toolbars need to implement this..
46801         //editor.on('editmodechange', this.updateToolbar, this);
46802     },
46803     
46804     
46805     
46806     /**
46807      * Protected method that will not generally be called directly. It triggers
46808      * a toolbar update by reading the markup state of the current selection in the editor.
46809      *
46810      * Note you can force an update by calling on('editorevent', scope, false)
46811      */
46812     updateToolbar: function(editor,ev,sel){
46813
46814         //Roo.log(ev);
46815         // capture mouse up - this is handy for selecting images..
46816         // perhaps should go somewhere else...
46817         if(!this.editorcore.activated){
46818              this.editor.onFirstFocus();
46819             return;
46820         }
46821         
46822         
46823         
46824         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46825         // selectNode - might want to handle IE?
46826         if (ev &&
46827             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46828             ev.target && ev.target.tagName == 'IMG') {
46829             // they have click on an image...
46830             // let's see if we can change the selection...
46831             sel = ev.target;
46832          
46833               var nodeRange = sel.ownerDocument.createRange();
46834             try {
46835                 nodeRange.selectNode(sel);
46836             } catch (e) {
46837                 nodeRange.selectNodeContents(sel);
46838             }
46839             //nodeRange.collapse(true);
46840             var s = this.editorcore.win.getSelection();
46841             s.removeAllRanges();
46842             s.addRange(nodeRange);
46843         }  
46844         
46845       
46846         var updateFooter = sel ? false : true;
46847         
46848         
46849         var ans = this.editorcore.getAllAncestors();
46850         
46851         // pick
46852         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46853         
46854         if (!sel) { 
46855             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46856             sel = sel ? sel : this.editorcore.doc.body;
46857             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46858             
46859         }
46860         // pick a menu that exists..
46861         var tn = sel.tagName.toUpperCase();
46862         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46863         
46864         tn = sel.tagName.toUpperCase();
46865         
46866         var lastSel = this.tb.selectedNode;
46867         
46868         this.tb.selectedNode = sel;
46869         
46870         // if current menu does not match..
46871         
46872         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46873                 
46874             this.tb.el.hide();
46875             ///console.log("show: " + tn);
46876             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46877             this.tb.el.show();
46878             // update name
46879             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46880             
46881             
46882             // update attributes
46883             if (this.tb.fields) {
46884                 this.tb.fields.each(function(e) {
46885                     if (e.stylename) {
46886                         e.setValue(sel.style[e.stylename]);
46887                         return;
46888                     } 
46889                    e.setValue(sel.getAttribute(e.attrname));
46890                 });
46891             }
46892             
46893             var hasStyles = false;
46894             for(var i in this.styles) {
46895                 hasStyles = true;
46896                 break;
46897             }
46898             
46899             // update styles
46900             if (hasStyles) { 
46901                 var st = this.tb.fields.item(0);
46902                 
46903                 st.store.removeAll();
46904                
46905                 
46906                 var cn = sel.className.split(/\s+/);
46907                 
46908                 var avs = [];
46909                 if (this.styles['*']) {
46910                     
46911                     Roo.each(this.styles['*'], function(v) {
46912                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46913                     });
46914                 }
46915                 if (this.styles[tn]) { 
46916                     Roo.each(this.styles[tn], function(v) {
46917                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46918                     });
46919                 }
46920                 
46921                 st.store.loadData(avs);
46922                 st.collapse();
46923                 st.setValue(cn);
46924             }
46925             // flag our selected Node.
46926             this.tb.selectedNode = sel;
46927            
46928            
46929             Roo.menu.MenuMgr.hideAll();
46930
46931         }
46932         
46933         if (!updateFooter) {
46934             //this.footDisp.dom.innerHTML = ''; 
46935             return;
46936         }
46937         // update the footer
46938         //
46939         var html = '';
46940         
46941         this.footerEls = ans.reverse();
46942         Roo.each(this.footerEls, function(a,i) {
46943             if (!a) { return; }
46944             html += html.length ? ' &gt; '  :  '';
46945             
46946             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46947             
46948         });
46949        
46950         // 
46951         var sz = this.footDisp.up('td').getSize();
46952         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46953         this.footDisp.dom.style.marginLeft = '5px';
46954         
46955         this.footDisp.dom.style.overflow = 'hidden';
46956         
46957         this.footDisp.dom.innerHTML = html;
46958             
46959         //this.editorsyncValue();
46960     },
46961      
46962     
46963    
46964        
46965     // private
46966     onDestroy : function(){
46967         if(this.rendered){
46968             
46969             this.tb.items.each(function(item){
46970                 if(item.menu){
46971                     item.menu.removeAll();
46972                     if(item.menu.el){
46973                         item.menu.el.destroy();
46974                     }
46975                 }
46976                 item.destroy();
46977             });
46978              
46979         }
46980     },
46981     onFirstFocus: function() {
46982         // need to do this for all the toolbars..
46983         this.tb.items.each(function(item){
46984            item.enable();
46985         });
46986     },
46987     buildToolbar: function(tlist, nm)
46988     {
46989         var editor = this.editor;
46990         var editorcore = this.editorcore;
46991          // create a new element.
46992         var wdiv = editor.wrap.createChild({
46993                 tag: 'div'
46994             }, editor.wrap.dom.firstChild.nextSibling, true);
46995         
46996        
46997         var tb = new Roo.Toolbar(wdiv);
46998         // add the name..
46999         
47000         tb.add(nm+ ":&nbsp;");
47001         
47002         var styles = [];
47003         for(var i in this.styles) {
47004             styles.push(i);
47005         }
47006         
47007         // styles...
47008         if (styles && styles.length) {
47009             
47010             // this needs a multi-select checkbox...
47011             tb.addField( new Roo.form.ComboBox({
47012                 store: new Roo.data.SimpleStore({
47013                     id : 'val',
47014                     fields: ['val', 'selected'],
47015                     data : [] 
47016                 }),
47017                 name : '-roo-edit-className',
47018                 attrname : 'className',
47019                 displayField: 'val',
47020                 typeAhead: false,
47021                 mode: 'local',
47022                 editable : false,
47023                 triggerAction: 'all',
47024                 emptyText:'Select Style',
47025                 selectOnFocus:true,
47026                 width: 130,
47027                 listeners : {
47028                     'select': function(c, r, i) {
47029                         // initial support only for on class per el..
47030                         tb.selectedNode.className =  r ? r.get('val') : '';
47031                         editorcore.syncValue();
47032                     }
47033                 }
47034     
47035             }));
47036         }
47037         
47038         var tbc = Roo.form.HtmlEditor.ToolbarContext;
47039         var tbops = tbc.options;
47040         
47041         for (var i in tlist) {
47042             
47043             var item = tlist[i];
47044             tb.add(item.title + ":&nbsp;");
47045             
47046             
47047             //optname == used so you can configure the options available..
47048             var opts = item.opts ? item.opts : false;
47049             if (item.optname) {
47050                 opts = tbops[item.optname];
47051            
47052             }
47053             
47054             if (opts) {
47055                 // opts == pulldown..
47056                 tb.addField( new Roo.form.ComboBox({
47057                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
47058                         id : 'val',
47059                         fields: ['val', 'display'],
47060                         data : opts  
47061                     }),
47062                     name : '-roo-edit-' + i,
47063                     attrname : i,
47064                     stylename : item.style ? item.style : false,
47065                     displayField: item.displayField ? item.displayField : 'val',
47066                     valueField :  'val',
47067                     typeAhead: false,
47068                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47069                     editable : false,
47070                     triggerAction: 'all',
47071                     emptyText:'Select',
47072                     selectOnFocus:true,
47073                     width: item.width ? item.width  : 130,
47074                     listeners : {
47075                         'select': function(c, r, i) {
47076                             if (c.stylename) {
47077                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47078                                 return;
47079                             }
47080                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47081                         }
47082                     }
47083
47084                 }));
47085                 continue;
47086                     
47087                  
47088                 
47089                 tb.addField( new Roo.form.TextField({
47090                     name: i,
47091                     width: 100,
47092                     //allowBlank:false,
47093                     value: ''
47094                 }));
47095                 continue;
47096             }
47097             tb.addField( new Roo.form.TextField({
47098                 name: '-roo-edit-' + i,
47099                 attrname : i,
47100                 
47101                 width: item.width,
47102                 //allowBlank:true,
47103                 value: '',
47104                 listeners: {
47105                     'change' : function(f, nv, ov) {
47106                         tb.selectedNode.setAttribute(f.attrname, nv);
47107                         editorcore.syncValue();
47108                     }
47109                 }
47110             }));
47111              
47112         }
47113         
47114         var _this = this;
47115         
47116         if(nm == 'BODY'){
47117             tb.addSeparator();
47118         
47119             tb.addButton( {
47120                 text: 'Stylesheets',
47121
47122                 listeners : {
47123                     click : function ()
47124                     {
47125                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47126                     }
47127                 }
47128             });
47129         }
47130         
47131         tb.addFill();
47132         tb.addButton( {
47133             text: 'Remove Tag',
47134     
47135             listeners : {
47136                 click : function ()
47137                 {
47138                     // remove
47139                     // undo does not work.
47140                      
47141                     var sn = tb.selectedNode;
47142                     
47143                     var pn = sn.parentNode;
47144                     
47145                     var stn =  sn.childNodes[0];
47146                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47147                     while (sn.childNodes.length) {
47148                         var node = sn.childNodes[0];
47149                         sn.removeChild(node);
47150                         //Roo.log(node);
47151                         pn.insertBefore(node, sn);
47152                         
47153                     }
47154                     pn.removeChild(sn);
47155                     var range = editorcore.createRange();
47156         
47157                     range.setStart(stn,0);
47158                     range.setEnd(en,0); //????
47159                     //range.selectNode(sel);
47160                     
47161                     
47162                     var selection = editorcore.getSelection();
47163                     selection.removeAllRanges();
47164                     selection.addRange(range);
47165                     
47166                     
47167                     
47168                     //_this.updateToolbar(null, null, pn);
47169                     _this.updateToolbar(null, null, null);
47170                     _this.footDisp.dom.innerHTML = ''; 
47171                 }
47172             }
47173             
47174                     
47175                 
47176             
47177         });
47178         
47179         
47180         tb.el.on('click', function(e){
47181             e.preventDefault(); // what does this do?
47182         });
47183         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47184         tb.el.hide();
47185         tb.name = nm;
47186         // dont need to disable them... as they will get hidden
47187         return tb;
47188          
47189         
47190     },
47191     buildFooter : function()
47192     {
47193         
47194         var fel = this.editor.wrap.createChild();
47195         this.footer = new Roo.Toolbar(fel);
47196         // toolbar has scrolly on left / right?
47197         var footDisp= new Roo.Toolbar.Fill();
47198         var _t = this;
47199         this.footer.add(
47200             {
47201                 text : '&lt;',
47202                 xtype: 'Button',
47203                 handler : function() {
47204                     _t.footDisp.scrollTo('left',0,true)
47205                 }
47206             }
47207         );
47208         this.footer.add( footDisp );
47209         this.footer.add( 
47210             {
47211                 text : '&gt;',
47212                 xtype: 'Button',
47213                 handler : function() {
47214                     // no animation..
47215                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47216                 }
47217             }
47218         );
47219         var fel = Roo.get(footDisp.el);
47220         fel.addClass('x-editor-context');
47221         this.footDispWrap = fel; 
47222         this.footDispWrap.overflow  = 'hidden';
47223         
47224         this.footDisp = fel.createChild();
47225         this.footDispWrap.on('click', this.onContextClick, this)
47226         
47227         
47228     },
47229     onContextClick : function (ev,dom)
47230     {
47231         ev.preventDefault();
47232         var  cn = dom.className;
47233         //Roo.log(cn);
47234         if (!cn.match(/x-ed-loc-/)) {
47235             return;
47236         }
47237         var n = cn.split('-').pop();
47238         var ans = this.footerEls;
47239         var sel = ans[n];
47240         
47241          // pick
47242         var range = this.editorcore.createRange();
47243         
47244         range.selectNodeContents(sel);
47245         //range.selectNode(sel);
47246         
47247         
47248         var selection = this.editorcore.getSelection();
47249         selection.removeAllRanges();
47250         selection.addRange(range);
47251         
47252         
47253         
47254         this.updateToolbar(null, null, sel);
47255         
47256         
47257     }
47258     
47259     
47260     
47261     
47262     
47263 });
47264
47265
47266
47267
47268
47269 /*
47270  * Based on:
47271  * Ext JS Library 1.1.1
47272  * Copyright(c) 2006-2007, Ext JS, LLC.
47273  *
47274  * Originally Released Under LGPL - original licence link has changed is not relivant.
47275  *
47276  * Fork - LGPL
47277  * <script type="text/javascript">
47278  */
47279  
47280 /**
47281  * @class Roo.form.BasicForm
47282  * @extends Roo.util.Observable
47283  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47284  * @constructor
47285  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47286  * @param {Object} config Configuration options
47287  */
47288 Roo.form.BasicForm = function(el, config){
47289     this.allItems = [];
47290     this.childForms = [];
47291     Roo.apply(this, config);
47292     /*
47293      * The Roo.form.Field items in this form.
47294      * @type MixedCollection
47295      */
47296      
47297      
47298     this.items = new Roo.util.MixedCollection(false, function(o){
47299         return o.id || (o.id = Roo.id());
47300     });
47301     this.addEvents({
47302         /**
47303          * @event beforeaction
47304          * Fires before any action is performed. Return false to cancel the action.
47305          * @param {Form} this
47306          * @param {Action} action The action to be performed
47307          */
47308         beforeaction: true,
47309         /**
47310          * @event actionfailed
47311          * Fires when an action fails.
47312          * @param {Form} this
47313          * @param {Action} action The action that failed
47314          */
47315         actionfailed : true,
47316         /**
47317          * @event actioncomplete
47318          * Fires when an action is completed.
47319          * @param {Form} this
47320          * @param {Action} action The action that completed
47321          */
47322         actioncomplete : true
47323     });
47324     if(el){
47325         this.initEl(el);
47326     }
47327     Roo.form.BasicForm.superclass.constructor.call(this);
47328     
47329     Roo.form.BasicForm.popover.apply();
47330 };
47331
47332 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47333     /**
47334      * @cfg {String} method
47335      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47336      */
47337     /**
47338      * @cfg {DataReader} reader
47339      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47340      * This is optional as there is built-in support for processing JSON.
47341      */
47342     /**
47343      * @cfg {DataReader} errorReader
47344      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47345      * This is completely optional as there is built-in support for processing JSON.
47346      */
47347     /**
47348      * @cfg {String} url
47349      * The URL to use for form actions if one isn't supplied in the action options.
47350      */
47351     /**
47352      * @cfg {Boolean} fileUpload
47353      * Set to true if this form is a file upload.
47354      */
47355      
47356     /**
47357      * @cfg {Object} baseParams
47358      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47359      */
47360      /**
47361      
47362     /**
47363      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47364      */
47365     timeout: 30,
47366
47367     // private
47368     activeAction : null,
47369
47370     /**
47371      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47372      * or setValues() data instead of when the form was first created.
47373      */
47374     trackResetOnLoad : false,
47375     
47376     
47377     /**
47378      * childForms - used for multi-tab forms
47379      * @type {Array}
47380      */
47381     childForms : false,
47382     
47383     /**
47384      * allItems - full list of fields.
47385      * @type {Array}
47386      */
47387     allItems : false,
47388     
47389     /**
47390      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47391      * element by passing it or its id or mask the form itself by passing in true.
47392      * @type Mixed
47393      */
47394     waitMsgTarget : false,
47395     
47396     /**
47397      * @type Boolean
47398      */
47399     disableMask : false,
47400     
47401     /**
47402      * @cfg {Boolean} errorMask (true|false) default false
47403      */
47404     errorMask : false,
47405     
47406     /**
47407      * @cfg {Number} maskOffset Default 100
47408      */
47409     maskOffset : 100,
47410
47411     // private
47412     initEl : function(el){
47413         this.el = Roo.get(el);
47414         this.id = this.el.id || Roo.id();
47415         this.el.on('submit', this.onSubmit, this);
47416         this.el.addClass('x-form');
47417     },
47418
47419     // private
47420     onSubmit : function(e){
47421         e.stopEvent();
47422     },
47423
47424     /**
47425      * Returns true if client-side validation on the form is successful.
47426      * @return Boolean
47427      */
47428     isValid : function(){
47429         var valid = true;
47430         var target = false;
47431         this.items.each(function(f){
47432             if(f.validate()){
47433                 return;
47434             }
47435             
47436             valid = false;
47437                 
47438             if(!target && f.el.isVisible(true)){
47439                 target = f;
47440             }
47441         });
47442         
47443         if(this.errorMask && !valid){
47444             Roo.form.BasicForm.popover.mask(this, target);
47445         }
47446         
47447         return valid;
47448     },
47449
47450     /**
47451      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47452      * @return Boolean
47453      */
47454     isDirty : function(){
47455         var dirty = false;
47456         this.items.each(function(f){
47457            if(f.isDirty()){
47458                dirty = true;
47459                return false;
47460            }
47461         });
47462         return dirty;
47463     },
47464     
47465     /**
47466      * Returns true if any fields in this form have changed since their original load. (New version)
47467      * @return Boolean
47468      */
47469     
47470     hasChanged : function()
47471     {
47472         var dirty = false;
47473         this.items.each(function(f){
47474            if(f.hasChanged()){
47475                dirty = true;
47476                return false;
47477            }
47478         });
47479         return dirty;
47480         
47481     },
47482     /**
47483      * Resets all hasChanged to 'false' -
47484      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47485      * So hasChanged storage is only to be used for this purpose
47486      * @return Boolean
47487      */
47488     resetHasChanged : function()
47489     {
47490         this.items.each(function(f){
47491            f.resetHasChanged();
47492         });
47493         
47494     },
47495     
47496     
47497     /**
47498      * Performs a predefined action (submit or load) or custom actions you define on this form.
47499      * @param {String} actionName The name of the action type
47500      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47501      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47502      * accept other config options):
47503      * <pre>
47504 Property          Type             Description
47505 ----------------  ---------------  ----------------------------------------------------------------------------------
47506 url               String           The url for the action (defaults to the form's url)
47507 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47508 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47509 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47510                                    validate the form on the client (defaults to false)
47511      * </pre>
47512      * @return {BasicForm} this
47513      */
47514     doAction : function(action, options){
47515         if(typeof action == 'string'){
47516             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47517         }
47518         if(this.fireEvent('beforeaction', this, action) !== false){
47519             this.beforeAction(action);
47520             action.run.defer(100, action);
47521         }
47522         return this;
47523     },
47524
47525     /**
47526      * Shortcut to do a submit action.
47527      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47528      * @return {BasicForm} this
47529      */
47530     submit : function(options){
47531         this.doAction('submit', options);
47532         return this;
47533     },
47534
47535     /**
47536      * Shortcut to do a load action.
47537      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47538      * @return {BasicForm} this
47539      */
47540     load : function(options){
47541         this.doAction('load', options);
47542         return this;
47543     },
47544
47545     /**
47546      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47547      * @param {Record} record The record to edit
47548      * @return {BasicForm} this
47549      */
47550     updateRecord : function(record){
47551         record.beginEdit();
47552         var fs = record.fields;
47553         fs.each(function(f){
47554             var field = this.findField(f.name);
47555             if(field){
47556                 record.set(f.name, field.getValue());
47557             }
47558         }, this);
47559         record.endEdit();
47560         return this;
47561     },
47562
47563     /**
47564      * Loads an Roo.data.Record into this form.
47565      * @param {Record} record The record to load
47566      * @return {BasicForm} this
47567      */
47568     loadRecord : function(record){
47569         this.setValues(record.data);
47570         return this;
47571     },
47572
47573     // private
47574     beforeAction : function(action){
47575         var o = action.options;
47576         
47577         if(!this.disableMask) {
47578             if(this.waitMsgTarget === true){
47579                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47580             }else if(this.waitMsgTarget){
47581                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47582                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47583             }else {
47584                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47585             }
47586         }
47587         
47588          
47589     },
47590
47591     // private
47592     afterAction : function(action, success){
47593         this.activeAction = null;
47594         var o = action.options;
47595         
47596         if(!this.disableMask) {
47597             if(this.waitMsgTarget === true){
47598                 this.el.unmask();
47599             }else if(this.waitMsgTarget){
47600                 this.waitMsgTarget.unmask();
47601             }else{
47602                 Roo.MessageBox.updateProgress(1);
47603                 Roo.MessageBox.hide();
47604             }
47605         }
47606         
47607         if(success){
47608             if(o.reset){
47609                 this.reset();
47610             }
47611             Roo.callback(o.success, o.scope, [this, action]);
47612             this.fireEvent('actioncomplete', this, action);
47613             
47614         }else{
47615             
47616             // failure condition..
47617             // we have a scenario where updates need confirming.
47618             // eg. if a locking scenario exists..
47619             // we look for { errors : { needs_confirm : true }} in the response.
47620             if (
47621                 (typeof(action.result) != 'undefined')  &&
47622                 (typeof(action.result.errors) != 'undefined')  &&
47623                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47624            ){
47625                 var _t = this;
47626                 Roo.MessageBox.confirm(
47627                     "Change requires confirmation",
47628                     action.result.errorMsg,
47629                     function(r) {
47630                         if (r != 'yes') {
47631                             return;
47632                         }
47633                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47634                     }
47635                     
47636                 );
47637                 
47638                 
47639                 
47640                 return;
47641             }
47642             
47643             Roo.callback(o.failure, o.scope, [this, action]);
47644             // show an error message if no failed handler is set..
47645             if (!this.hasListener('actionfailed')) {
47646                 Roo.MessageBox.alert("Error",
47647                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47648                         action.result.errorMsg :
47649                         "Saving Failed, please check your entries or try again"
47650                 );
47651             }
47652             
47653             this.fireEvent('actionfailed', this, action);
47654         }
47655         
47656     },
47657
47658     /**
47659      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47660      * @param {String} id The value to search for
47661      * @return Field
47662      */
47663     findField : function(id){
47664         var field = this.items.get(id);
47665         if(!field){
47666             this.items.each(function(f){
47667                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47668                     field = f;
47669                     return false;
47670                 }
47671             });
47672         }
47673         return field || null;
47674     },
47675
47676     /**
47677      * Add a secondary form to this one, 
47678      * Used to provide tabbed forms. One form is primary, with hidden values 
47679      * which mirror the elements from the other forms.
47680      * 
47681      * @param {Roo.form.Form} form to add.
47682      * 
47683      */
47684     addForm : function(form)
47685     {
47686        
47687         if (this.childForms.indexOf(form) > -1) {
47688             // already added..
47689             return;
47690         }
47691         this.childForms.push(form);
47692         var n = '';
47693         Roo.each(form.allItems, function (fe) {
47694             
47695             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47696             if (this.findField(n)) { // already added..
47697                 return;
47698             }
47699             var add = new Roo.form.Hidden({
47700                 name : n
47701             });
47702             add.render(this.el);
47703             
47704             this.add( add );
47705         }, this);
47706         
47707     },
47708     /**
47709      * Mark fields in this form invalid in bulk.
47710      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47711      * @return {BasicForm} this
47712      */
47713     markInvalid : function(errors){
47714         if(errors instanceof Array){
47715             for(var i = 0, len = errors.length; i < len; i++){
47716                 var fieldError = errors[i];
47717                 var f = this.findField(fieldError.id);
47718                 if(f){
47719                     f.markInvalid(fieldError.msg);
47720                 }
47721             }
47722         }else{
47723             var field, id;
47724             for(id in errors){
47725                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47726                     field.markInvalid(errors[id]);
47727                 }
47728             }
47729         }
47730         Roo.each(this.childForms || [], function (f) {
47731             f.markInvalid(errors);
47732         });
47733         
47734         return this;
47735     },
47736
47737     /**
47738      * Set values for fields in this form in bulk.
47739      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47740      * @return {BasicForm} this
47741      */
47742     setValues : function(values){
47743         if(values instanceof Array){ // array of objects
47744             for(var i = 0, len = values.length; i < len; i++){
47745                 var v = values[i];
47746                 var f = this.findField(v.id);
47747                 if(f){
47748                     f.setValue(v.value);
47749                     if(this.trackResetOnLoad){
47750                         f.originalValue = f.getValue();
47751                     }
47752                 }
47753             }
47754         }else{ // object hash
47755             var field, id;
47756             for(id in values){
47757                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47758                     
47759                     if (field.setFromData && 
47760                         field.valueField && 
47761                         field.displayField &&
47762                         // combos' with local stores can 
47763                         // be queried via setValue()
47764                         // to set their value..
47765                         (field.store && !field.store.isLocal)
47766                         ) {
47767                         // it's a combo
47768                         var sd = { };
47769                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47770                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47771                         field.setFromData(sd);
47772                         
47773                     } else {
47774                         field.setValue(values[id]);
47775                     }
47776                     
47777                     
47778                     if(this.trackResetOnLoad){
47779                         field.originalValue = field.getValue();
47780                     }
47781                 }
47782             }
47783         }
47784         this.resetHasChanged();
47785         
47786         
47787         Roo.each(this.childForms || [], function (f) {
47788             f.setValues(values);
47789             f.resetHasChanged();
47790         });
47791                 
47792         return this;
47793     },
47794  
47795     /**
47796      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47797      * they are returned as an array.
47798      * @param {Boolean} asString
47799      * @return {Object}
47800      */
47801     getValues : function(asString){
47802         if (this.childForms) {
47803             // copy values from the child forms
47804             Roo.each(this.childForms, function (f) {
47805                 this.setValues(f.getValues());
47806             }, this);
47807         }
47808         
47809         // use formdata
47810         if (typeof(FormData) != 'undefined' && asString !== true) {
47811             var fd = (new FormData(this.el.dom)).entries();
47812             var ret = {};
47813             var ent = fd.next();
47814             while (!ent.done) {
47815                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47816                 ent = fd.next();
47817             };
47818             return ret;
47819         }
47820         
47821         
47822         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47823         if(asString === true){
47824             return fs;
47825         }
47826         return Roo.urlDecode(fs);
47827     },
47828     
47829     /**
47830      * Returns the fields in this form as an object with key/value pairs. 
47831      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47832      * @return {Object}
47833      */
47834     getFieldValues : function(with_hidden)
47835     {
47836         if (this.childForms) {
47837             // copy values from the child forms
47838             // should this call getFieldValues - probably not as we do not currently copy
47839             // hidden fields when we generate..
47840             Roo.each(this.childForms, function (f) {
47841                 this.setValues(f.getValues());
47842             }, this);
47843         }
47844         
47845         var ret = {};
47846         this.items.each(function(f){
47847             if (!f.getName()) {
47848                 return;
47849             }
47850             var v = f.getValue();
47851             if (f.inputType =='radio') {
47852                 if (typeof(ret[f.getName()]) == 'undefined') {
47853                     ret[f.getName()] = ''; // empty..
47854                 }
47855                 
47856                 if (!f.el.dom.checked) {
47857                     return;
47858                     
47859                 }
47860                 v = f.el.dom.value;
47861                 
47862             }
47863             
47864             // not sure if this supported any more..
47865             if ((typeof(v) == 'object') && f.getRawValue) {
47866                 v = f.getRawValue() ; // dates..
47867             }
47868             // combo boxes where name != hiddenName...
47869             if (f.name != f.getName()) {
47870                 ret[f.name] = f.getRawValue();
47871             }
47872             ret[f.getName()] = v;
47873         });
47874         
47875         return ret;
47876     },
47877
47878     /**
47879      * Clears all invalid messages in this form.
47880      * @return {BasicForm} this
47881      */
47882     clearInvalid : function(){
47883         this.items.each(function(f){
47884            f.clearInvalid();
47885         });
47886         
47887         Roo.each(this.childForms || [], function (f) {
47888             f.clearInvalid();
47889         });
47890         
47891         
47892         return this;
47893     },
47894
47895     /**
47896      * Resets this form.
47897      * @return {BasicForm} this
47898      */
47899     reset : function(){
47900         this.items.each(function(f){
47901             f.reset();
47902         });
47903         
47904         Roo.each(this.childForms || [], function (f) {
47905             f.reset();
47906         });
47907         this.resetHasChanged();
47908         
47909         return this;
47910     },
47911
47912     /**
47913      * Add Roo.form components to this form.
47914      * @param {Field} field1
47915      * @param {Field} field2 (optional)
47916      * @param {Field} etc (optional)
47917      * @return {BasicForm} this
47918      */
47919     add : function(){
47920         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47921         return this;
47922     },
47923
47924
47925     /**
47926      * Removes a field from the items collection (does NOT remove its markup).
47927      * @param {Field} field
47928      * @return {BasicForm} this
47929      */
47930     remove : function(field){
47931         this.items.remove(field);
47932         return this;
47933     },
47934
47935     /**
47936      * Looks at the fields in this form, checks them for an id attribute,
47937      * and calls applyTo on the existing dom element with that id.
47938      * @return {BasicForm} this
47939      */
47940     render : function(){
47941         this.items.each(function(f){
47942             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47943                 f.applyTo(f.id);
47944             }
47945         });
47946         return this;
47947     },
47948
47949     /**
47950      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47951      * @param {Object} values
47952      * @return {BasicForm} this
47953      */
47954     applyToFields : function(o){
47955         this.items.each(function(f){
47956            Roo.apply(f, o);
47957         });
47958         return this;
47959     },
47960
47961     /**
47962      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47963      * @param {Object} values
47964      * @return {BasicForm} this
47965      */
47966     applyIfToFields : function(o){
47967         this.items.each(function(f){
47968            Roo.applyIf(f, o);
47969         });
47970         return this;
47971     }
47972 });
47973
47974 // back compat
47975 Roo.BasicForm = Roo.form.BasicForm;
47976
47977 Roo.apply(Roo.form.BasicForm, {
47978     
47979     popover : {
47980         
47981         padding : 5,
47982         
47983         isApplied : false,
47984         
47985         isMasked : false,
47986         
47987         form : false,
47988         
47989         target : false,
47990         
47991         intervalID : false,
47992         
47993         maskEl : false,
47994         
47995         apply : function()
47996         {
47997             if(this.isApplied){
47998                 return;
47999             }
48000             
48001             this.maskEl = {
48002                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
48003                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
48004                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
48005                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
48006             };
48007             
48008             this.maskEl.top.enableDisplayMode("block");
48009             this.maskEl.left.enableDisplayMode("block");
48010             this.maskEl.bottom.enableDisplayMode("block");
48011             this.maskEl.right.enableDisplayMode("block");
48012             
48013             Roo.get(document.body).on('click', function(){
48014                 this.unmask();
48015             }, this);
48016             
48017             Roo.get(document.body).on('touchstart', function(){
48018                 this.unmask();
48019             }, this);
48020             
48021             this.isApplied = true
48022         },
48023         
48024         mask : function(form, target)
48025         {
48026             this.form = form;
48027             
48028             this.target = target;
48029             
48030             if(!this.form.errorMask || !target.el){
48031                 return;
48032             }
48033             
48034             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
48035             
48036             var ot = this.target.el.calcOffsetsTo(scrollable);
48037             
48038             var scrollTo = ot[1] - this.form.maskOffset;
48039             
48040             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
48041             
48042             scrollable.scrollTo('top', scrollTo);
48043             
48044             var el = this.target.wrap || this.target.el;
48045             
48046             var box = el.getBox();
48047             
48048             this.maskEl.top.setStyle('position', 'absolute');
48049             this.maskEl.top.setStyle('z-index', 10000);
48050             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
48051             this.maskEl.top.setLeft(0);
48052             this.maskEl.top.setTop(0);
48053             this.maskEl.top.show();
48054             
48055             this.maskEl.left.setStyle('position', 'absolute');
48056             this.maskEl.left.setStyle('z-index', 10000);
48057             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
48058             this.maskEl.left.setLeft(0);
48059             this.maskEl.left.setTop(box.y - this.padding);
48060             this.maskEl.left.show();
48061
48062             this.maskEl.bottom.setStyle('position', 'absolute');
48063             this.maskEl.bottom.setStyle('z-index', 10000);
48064             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48065             this.maskEl.bottom.setLeft(0);
48066             this.maskEl.bottom.setTop(box.bottom + this.padding);
48067             this.maskEl.bottom.show();
48068
48069             this.maskEl.right.setStyle('position', 'absolute');
48070             this.maskEl.right.setStyle('z-index', 10000);
48071             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48072             this.maskEl.right.setLeft(box.right + this.padding);
48073             this.maskEl.right.setTop(box.y - this.padding);
48074             this.maskEl.right.show();
48075
48076             this.intervalID = window.setInterval(function() {
48077                 Roo.form.BasicForm.popover.unmask();
48078             }, 10000);
48079
48080             window.onwheel = function(){ return false;};
48081             
48082             (function(){ this.isMasked = true; }).defer(500, this);
48083             
48084         },
48085         
48086         unmask : function()
48087         {
48088             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48089                 return;
48090             }
48091             
48092             this.maskEl.top.setStyle('position', 'absolute');
48093             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48094             this.maskEl.top.hide();
48095
48096             this.maskEl.left.setStyle('position', 'absolute');
48097             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48098             this.maskEl.left.hide();
48099
48100             this.maskEl.bottom.setStyle('position', 'absolute');
48101             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48102             this.maskEl.bottom.hide();
48103
48104             this.maskEl.right.setStyle('position', 'absolute');
48105             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48106             this.maskEl.right.hide();
48107             
48108             window.onwheel = function(){ return true;};
48109             
48110             if(this.intervalID){
48111                 window.clearInterval(this.intervalID);
48112                 this.intervalID = false;
48113             }
48114             
48115             this.isMasked = false;
48116             
48117         }
48118         
48119     }
48120     
48121 });/*
48122  * Based on:
48123  * Ext JS Library 1.1.1
48124  * Copyright(c) 2006-2007, Ext JS, LLC.
48125  *
48126  * Originally Released Under LGPL - original licence link has changed is not relivant.
48127  *
48128  * Fork - LGPL
48129  * <script type="text/javascript">
48130  */
48131
48132 /**
48133  * @class Roo.form.Form
48134  * @extends Roo.form.BasicForm
48135  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48136  * @constructor
48137  * @param {Object} config Configuration options
48138  */
48139 Roo.form.Form = function(config){
48140     var xitems =  [];
48141     if (config.items) {
48142         xitems = config.items;
48143         delete config.items;
48144     }
48145    
48146     
48147     Roo.form.Form.superclass.constructor.call(this, null, config);
48148     this.url = this.url || this.action;
48149     if(!this.root){
48150         this.root = new Roo.form.Layout(Roo.applyIf({
48151             id: Roo.id()
48152         }, config));
48153     }
48154     this.active = this.root;
48155     /**
48156      * Array of all the buttons that have been added to this form via {@link addButton}
48157      * @type Array
48158      */
48159     this.buttons = [];
48160     this.allItems = [];
48161     this.addEvents({
48162         /**
48163          * @event clientvalidation
48164          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48165          * @param {Form} this
48166          * @param {Boolean} valid true if the form has passed client-side validation
48167          */
48168         clientvalidation: true,
48169         /**
48170          * @event rendered
48171          * Fires when the form is rendered
48172          * @param {Roo.form.Form} form
48173          */
48174         rendered : true
48175     });
48176     
48177     if (this.progressUrl) {
48178             // push a hidden field onto the list of fields..
48179             this.addxtype( {
48180                     xns: Roo.form, 
48181                     xtype : 'Hidden', 
48182                     name : 'UPLOAD_IDENTIFIER' 
48183             });
48184         }
48185         
48186     
48187     Roo.each(xitems, this.addxtype, this);
48188     
48189 };
48190
48191 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48192     /**
48193      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48194      */
48195     /**
48196      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48197      */
48198     /**
48199      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48200      */
48201     buttonAlign:'center',
48202
48203     /**
48204      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48205      */
48206     minButtonWidth:75,
48207
48208     /**
48209      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48210      * This property cascades to child containers if not set.
48211      */
48212     labelAlign:'left',
48213
48214     /**
48215      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48216      * fires a looping event with that state. This is required to bind buttons to the valid
48217      * state using the config value formBind:true on the button.
48218      */
48219     monitorValid : false,
48220
48221     /**
48222      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48223      */
48224     monitorPoll : 200,
48225     
48226     /**
48227      * @cfg {String} progressUrl - Url to return progress data 
48228      */
48229     
48230     progressUrl : false,
48231     /**
48232      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48233      * sending a formdata with extra parameters - eg uploaded elements.
48234      */
48235     
48236     formData : false,
48237     
48238     /**
48239      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48240      * fields are added and the column is closed. If no fields are passed the column remains open
48241      * until end() is called.
48242      * @param {Object} config The config to pass to the column
48243      * @param {Field} field1 (optional)
48244      * @param {Field} field2 (optional)
48245      * @param {Field} etc (optional)
48246      * @return Column The column container object
48247      */
48248     column : function(c){
48249         var col = new Roo.form.Column(c);
48250         this.start(col);
48251         if(arguments.length > 1){ // duplicate code required because of Opera
48252             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48253             this.end();
48254         }
48255         return col;
48256     },
48257
48258     /**
48259      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48260      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48261      * until end() is called.
48262      * @param {Object} config The config to pass to the fieldset
48263      * @param {Field} field1 (optional)
48264      * @param {Field} field2 (optional)
48265      * @param {Field} etc (optional)
48266      * @return FieldSet The fieldset container object
48267      */
48268     fieldset : function(c){
48269         var fs = new Roo.form.FieldSet(c);
48270         this.start(fs);
48271         if(arguments.length > 1){ // duplicate code required because of Opera
48272             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48273             this.end();
48274         }
48275         return fs;
48276     },
48277
48278     /**
48279      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48280      * fields are added and the container is closed. If no fields are passed the container remains open
48281      * until end() is called.
48282      * @param {Object} config The config to pass to the Layout
48283      * @param {Field} field1 (optional)
48284      * @param {Field} field2 (optional)
48285      * @param {Field} etc (optional)
48286      * @return Layout The container object
48287      */
48288     container : function(c){
48289         var l = new Roo.form.Layout(c);
48290         this.start(l);
48291         if(arguments.length > 1){ // duplicate code required because of Opera
48292             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48293             this.end();
48294         }
48295         return l;
48296     },
48297
48298     /**
48299      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48300      * @param {Object} container A Roo.form.Layout or subclass of Layout
48301      * @return {Form} this
48302      */
48303     start : function(c){
48304         // cascade label info
48305         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48306         this.active.stack.push(c);
48307         c.ownerCt = this.active;
48308         this.active = c;
48309         return this;
48310     },
48311
48312     /**
48313      * Closes the current open container
48314      * @return {Form} this
48315      */
48316     end : function(){
48317         if(this.active == this.root){
48318             return this;
48319         }
48320         this.active = this.active.ownerCt;
48321         return this;
48322     },
48323
48324     /**
48325      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48326      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48327      * as the label of the field.
48328      * @param {Field} field1
48329      * @param {Field} field2 (optional)
48330      * @param {Field} etc. (optional)
48331      * @return {Form} this
48332      */
48333     add : function(){
48334         this.active.stack.push.apply(this.active.stack, arguments);
48335         this.allItems.push.apply(this.allItems,arguments);
48336         var r = [];
48337         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48338             if(a[i].isFormField){
48339                 r.push(a[i]);
48340             }
48341         }
48342         if(r.length > 0){
48343             Roo.form.Form.superclass.add.apply(this, r);
48344         }
48345         return this;
48346     },
48347     
48348
48349     
48350     
48351     
48352      /**
48353      * Find any element that has been added to a form, using it's ID or name
48354      * This can include framesets, columns etc. along with regular fields..
48355      * @param {String} id - id or name to find.
48356      
48357      * @return {Element} e - or false if nothing found.
48358      */
48359     findbyId : function(id)
48360     {
48361         var ret = false;
48362         if (!id) {
48363             return ret;
48364         }
48365         Roo.each(this.allItems, function(f){
48366             if (f.id == id || f.name == id ){
48367                 ret = f;
48368                 return false;
48369             }
48370         });
48371         return ret;
48372     },
48373
48374     
48375     
48376     /**
48377      * Render this form into the passed container. This should only be called once!
48378      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48379      * @return {Form} this
48380      */
48381     render : function(ct)
48382     {
48383         
48384         
48385         
48386         ct = Roo.get(ct);
48387         var o = this.autoCreate || {
48388             tag: 'form',
48389             method : this.method || 'POST',
48390             id : this.id || Roo.id()
48391         };
48392         this.initEl(ct.createChild(o));
48393
48394         this.root.render(this.el);
48395         
48396        
48397              
48398         this.items.each(function(f){
48399             f.render('x-form-el-'+f.id);
48400         });
48401
48402         if(this.buttons.length > 0){
48403             // tables are required to maintain order and for correct IE layout
48404             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48405                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48406                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48407             }}, null, true);
48408             var tr = tb.getElementsByTagName('tr')[0];
48409             for(var i = 0, len = this.buttons.length; i < len; i++) {
48410                 var b = this.buttons[i];
48411                 var td = document.createElement('td');
48412                 td.className = 'x-form-btn-td';
48413                 b.render(tr.appendChild(td));
48414             }
48415         }
48416         if(this.monitorValid){ // initialize after render
48417             this.startMonitoring();
48418         }
48419         this.fireEvent('rendered', this);
48420         return this;
48421     },
48422
48423     /**
48424      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48425      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48426      * object or a valid Roo.DomHelper element config
48427      * @param {Function} handler The function called when the button is clicked
48428      * @param {Object} scope (optional) The scope of the handler function
48429      * @return {Roo.Button}
48430      */
48431     addButton : function(config, handler, scope){
48432         var bc = {
48433             handler: handler,
48434             scope: scope,
48435             minWidth: this.minButtonWidth,
48436             hideParent:true
48437         };
48438         if(typeof config == "string"){
48439             bc.text = config;
48440         }else{
48441             Roo.apply(bc, config);
48442         }
48443         var btn = new Roo.Button(null, bc);
48444         this.buttons.push(btn);
48445         return btn;
48446     },
48447
48448      /**
48449      * Adds a series of form elements (using the xtype property as the factory method.
48450      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48451      * @param {Object} config 
48452      */
48453     
48454     addxtype : function()
48455     {
48456         var ar = Array.prototype.slice.call(arguments, 0);
48457         var ret = false;
48458         for(var i = 0; i < ar.length; i++) {
48459             if (!ar[i]) {
48460                 continue; // skip -- if this happends something invalid got sent, we 
48461                 // should ignore it, as basically that interface element will not show up
48462                 // and that should be pretty obvious!!
48463             }
48464             
48465             if (Roo.form[ar[i].xtype]) {
48466                 ar[i].form = this;
48467                 var fe = Roo.factory(ar[i], Roo.form);
48468                 if (!ret) {
48469                     ret = fe;
48470                 }
48471                 fe.form = this;
48472                 if (fe.store) {
48473                     fe.store.form = this;
48474                 }
48475                 if (fe.isLayout) {  
48476                          
48477                     this.start(fe);
48478                     this.allItems.push(fe);
48479                     if (fe.items && fe.addxtype) {
48480                         fe.addxtype.apply(fe, fe.items);
48481                         delete fe.items;
48482                     }
48483                      this.end();
48484                     continue;
48485                 }
48486                 
48487                 
48488                  
48489                 this.add(fe);
48490               //  console.log('adding ' + ar[i].xtype);
48491             }
48492             if (ar[i].xtype == 'Button') {  
48493                 //console.log('adding button');
48494                 //console.log(ar[i]);
48495                 this.addButton(ar[i]);
48496                 this.allItems.push(fe);
48497                 continue;
48498             }
48499             
48500             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48501                 alert('end is not supported on xtype any more, use items');
48502             //    this.end();
48503             //    //console.log('adding end');
48504             }
48505             
48506         }
48507         return ret;
48508     },
48509     
48510     /**
48511      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48512      * option "monitorValid"
48513      */
48514     startMonitoring : function(){
48515         if(!this.bound){
48516             this.bound = true;
48517             Roo.TaskMgr.start({
48518                 run : this.bindHandler,
48519                 interval : this.monitorPoll || 200,
48520                 scope: this
48521             });
48522         }
48523     },
48524
48525     /**
48526      * Stops monitoring of the valid state of this form
48527      */
48528     stopMonitoring : function(){
48529         this.bound = false;
48530     },
48531
48532     // private
48533     bindHandler : function(){
48534         if(!this.bound){
48535             return false; // stops binding
48536         }
48537         var valid = true;
48538         this.items.each(function(f){
48539             if(!f.isValid(true)){
48540                 valid = false;
48541                 return false;
48542             }
48543         });
48544         for(var i = 0, len = this.buttons.length; i < len; i++){
48545             var btn = this.buttons[i];
48546             if(btn.formBind === true && btn.disabled === valid){
48547                 btn.setDisabled(!valid);
48548             }
48549         }
48550         this.fireEvent('clientvalidation', this, valid);
48551     }
48552     
48553     
48554     
48555     
48556     
48557     
48558     
48559     
48560 });
48561
48562
48563 // back compat
48564 Roo.Form = Roo.form.Form;
48565 /*
48566  * Based on:
48567  * Ext JS Library 1.1.1
48568  * Copyright(c) 2006-2007, Ext JS, LLC.
48569  *
48570  * Originally Released Under LGPL - original licence link has changed is not relivant.
48571  *
48572  * Fork - LGPL
48573  * <script type="text/javascript">
48574  */
48575
48576 // as we use this in bootstrap.
48577 Roo.namespace('Roo.form');
48578  /**
48579  * @class Roo.form.Action
48580  * Internal Class used to handle form actions
48581  * @constructor
48582  * @param {Roo.form.BasicForm} el The form element or its id
48583  * @param {Object} config Configuration options
48584  */
48585
48586  
48587  
48588 // define the action interface
48589 Roo.form.Action = function(form, options){
48590     this.form = form;
48591     this.options = options || {};
48592 };
48593 /**
48594  * Client Validation Failed
48595  * @const 
48596  */
48597 Roo.form.Action.CLIENT_INVALID = 'client';
48598 /**
48599  * Server Validation Failed
48600  * @const 
48601  */
48602 Roo.form.Action.SERVER_INVALID = 'server';
48603  /**
48604  * Connect to Server Failed
48605  * @const 
48606  */
48607 Roo.form.Action.CONNECT_FAILURE = 'connect';
48608 /**
48609  * Reading Data from Server Failed
48610  * @const 
48611  */
48612 Roo.form.Action.LOAD_FAILURE = 'load';
48613
48614 Roo.form.Action.prototype = {
48615     type : 'default',
48616     failureType : undefined,
48617     response : undefined,
48618     result : undefined,
48619
48620     // interface method
48621     run : function(options){
48622
48623     },
48624
48625     // interface method
48626     success : function(response){
48627
48628     },
48629
48630     // interface method
48631     handleResponse : function(response){
48632
48633     },
48634
48635     // default connection failure
48636     failure : function(response){
48637         
48638         this.response = response;
48639         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48640         this.form.afterAction(this, false);
48641     },
48642
48643     processResponse : function(response){
48644         this.response = response;
48645         if(!response.responseText){
48646             return true;
48647         }
48648         this.result = this.handleResponse(response);
48649         return this.result;
48650     },
48651
48652     // utility functions used internally
48653     getUrl : function(appendParams){
48654         var url = this.options.url || this.form.url || this.form.el.dom.action;
48655         if(appendParams){
48656             var p = this.getParams();
48657             if(p){
48658                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48659             }
48660         }
48661         return url;
48662     },
48663
48664     getMethod : function(){
48665         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48666     },
48667
48668     getParams : function(){
48669         var bp = this.form.baseParams;
48670         var p = this.options.params;
48671         if(p){
48672             if(typeof p == "object"){
48673                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48674             }else if(typeof p == 'string' && bp){
48675                 p += '&' + Roo.urlEncode(bp);
48676             }
48677         }else if(bp){
48678             p = Roo.urlEncode(bp);
48679         }
48680         return p;
48681     },
48682
48683     createCallback : function(){
48684         return {
48685             success: this.success,
48686             failure: this.failure,
48687             scope: this,
48688             timeout: (this.form.timeout*1000),
48689             upload: this.form.fileUpload ? this.success : undefined
48690         };
48691     }
48692 };
48693
48694 Roo.form.Action.Submit = function(form, options){
48695     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48696 };
48697
48698 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48699     type : 'submit',
48700
48701     haveProgress : false,
48702     uploadComplete : false,
48703     
48704     // uploadProgress indicator.
48705     uploadProgress : function()
48706     {
48707         if (!this.form.progressUrl) {
48708             return;
48709         }
48710         
48711         if (!this.haveProgress) {
48712             Roo.MessageBox.progress("Uploading", "Uploading");
48713         }
48714         if (this.uploadComplete) {
48715            Roo.MessageBox.hide();
48716            return;
48717         }
48718         
48719         this.haveProgress = true;
48720    
48721         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48722         
48723         var c = new Roo.data.Connection();
48724         c.request({
48725             url : this.form.progressUrl,
48726             params: {
48727                 id : uid
48728             },
48729             method: 'GET',
48730             success : function(req){
48731                //console.log(data);
48732                 var rdata = false;
48733                 var edata;
48734                 try  {
48735                    rdata = Roo.decode(req.responseText)
48736                 } catch (e) {
48737                     Roo.log("Invalid data from server..");
48738                     Roo.log(edata);
48739                     return;
48740                 }
48741                 if (!rdata || !rdata.success) {
48742                     Roo.log(rdata);
48743                     Roo.MessageBox.alert(Roo.encode(rdata));
48744                     return;
48745                 }
48746                 var data = rdata.data;
48747                 
48748                 if (this.uploadComplete) {
48749                    Roo.MessageBox.hide();
48750                    return;
48751                 }
48752                    
48753                 if (data){
48754                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48755                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48756                     );
48757                 }
48758                 this.uploadProgress.defer(2000,this);
48759             },
48760        
48761             failure: function(data) {
48762                 Roo.log('progress url failed ');
48763                 Roo.log(data);
48764             },
48765             scope : this
48766         });
48767            
48768     },
48769     
48770     
48771     run : function()
48772     {
48773         // run get Values on the form, so it syncs any secondary forms.
48774         this.form.getValues();
48775         
48776         var o = this.options;
48777         var method = this.getMethod();
48778         var isPost = method == 'POST';
48779         if(o.clientValidation === false || this.form.isValid()){
48780             
48781             if (this.form.progressUrl) {
48782                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48783                     (new Date() * 1) + '' + Math.random());
48784                     
48785             } 
48786             
48787             
48788             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48789                 form:this.form.el.dom,
48790                 url:this.getUrl(!isPost),
48791                 method: method,
48792                 params:isPost ? this.getParams() : null,
48793                 isUpload: this.form.fileUpload,
48794                 formData : this.form.formData
48795             }));
48796             
48797             this.uploadProgress();
48798
48799         }else if (o.clientValidation !== false){ // client validation failed
48800             this.failureType = Roo.form.Action.CLIENT_INVALID;
48801             this.form.afterAction(this, false);
48802         }
48803     },
48804
48805     success : function(response)
48806     {
48807         this.uploadComplete= true;
48808         if (this.haveProgress) {
48809             Roo.MessageBox.hide();
48810         }
48811         
48812         
48813         var result = this.processResponse(response);
48814         if(result === true || result.success){
48815             this.form.afterAction(this, true);
48816             return;
48817         }
48818         if(result.errors){
48819             this.form.markInvalid(result.errors);
48820             this.failureType = Roo.form.Action.SERVER_INVALID;
48821         }
48822         this.form.afterAction(this, false);
48823     },
48824     failure : function(response)
48825     {
48826         this.uploadComplete= true;
48827         if (this.haveProgress) {
48828             Roo.MessageBox.hide();
48829         }
48830         
48831         this.response = response;
48832         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48833         this.form.afterAction(this, false);
48834     },
48835     
48836     handleResponse : function(response){
48837         if(this.form.errorReader){
48838             var rs = this.form.errorReader.read(response);
48839             var errors = [];
48840             if(rs.records){
48841                 for(var i = 0, len = rs.records.length; i < len; i++) {
48842                     var r = rs.records[i];
48843                     errors[i] = r.data;
48844                 }
48845             }
48846             if(errors.length < 1){
48847                 errors = null;
48848             }
48849             return {
48850                 success : rs.success,
48851                 errors : errors
48852             };
48853         }
48854         var ret = false;
48855         try {
48856             ret = Roo.decode(response.responseText);
48857         } catch (e) {
48858             ret = {
48859                 success: false,
48860                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48861                 errors : []
48862             };
48863         }
48864         return ret;
48865         
48866     }
48867 });
48868
48869
48870 Roo.form.Action.Load = function(form, options){
48871     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48872     this.reader = this.form.reader;
48873 };
48874
48875 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48876     type : 'load',
48877
48878     run : function(){
48879         
48880         Roo.Ajax.request(Roo.apply(
48881                 this.createCallback(), {
48882                     method:this.getMethod(),
48883                     url:this.getUrl(false),
48884                     params:this.getParams()
48885         }));
48886     },
48887
48888     success : function(response){
48889         
48890         var result = this.processResponse(response);
48891         if(result === true || !result.success || !result.data){
48892             this.failureType = Roo.form.Action.LOAD_FAILURE;
48893             this.form.afterAction(this, false);
48894             return;
48895         }
48896         this.form.clearInvalid();
48897         this.form.setValues(result.data);
48898         this.form.afterAction(this, true);
48899     },
48900
48901     handleResponse : function(response){
48902         if(this.form.reader){
48903             var rs = this.form.reader.read(response);
48904             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48905             return {
48906                 success : rs.success,
48907                 data : data
48908             };
48909         }
48910         return Roo.decode(response.responseText);
48911     }
48912 });
48913
48914 Roo.form.Action.ACTION_TYPES = {
48915     'load' : Roo.form.Action.Load,
48916     'submit' : Roo.form.Action.Submit
48917 };/*
48918  * Based on:
48919  * Ext JS Library 1.1.1
48920  * Copyright(c) 2006-2007, Ext JS, LLC.
48921  *
48922  * Originally Released Under LGPL - original licence link has changed is not relivant.
48923  *
48924  * Fork - LGPL
48925  * <script type="text/javascript">
48926  */
48927  
48928 /**
48929  * @class Roo.form.Layout
48930  * @extends Roo.Component
48931  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48932  * @constructor
48933  * @param {Object} config Configuration options
48934  */
48935 Roo.form.Layout = function(config){
48936     var xitems = [];
48937     if (config.items) {
48938         xitems = config.items;
48939         delete config.items;
48940     }
48941     Roo.form.Layout.superclass.constructor.call(this, config);
48942     this.stack = [];
48943     Roo.each(xitems, this.addxtype, this);
48944      
48945 };
48946
48947 Roo.extend(Roo.form.Layout, Roo.Component, {
48948     /**
48949      * @cfg {String/Object} autoCreate
48950      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48951      */
48952     /**
48953      * @cfg {String/Object/Function} style
48954      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48955      * a function which returns such a specification.
48956      */
48957     /**
48958      * @cfg {String} labelAlign
48959      * Valid values are "left," "top" and "right" (defaults to "left")
48960      */
48961     /**
48962      * @cfg {Number} labelWidth
48963      * Fixed width in pixels of all field labels (defaults to undefined)
48964      */
48965     /**
48966      * @cfg {Boolean} clear
48967      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48968      */
48969     clear : true,
48970     /**
48971      * @cfg {String} labelSeparator
48972      * The separator to use after field labels (defaults to ':')
48973      */
48974     labelSeparator : ':',
48975     /**
48976      * @cfg {Boolean} hideLabels
48977      * True to suppress the display of field labels in this layout (defaults to false)
48978      */
48979     hideLabels : false,
48980
48981     // private
48982     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48983     
48984     isLayout : true,
48985     
48986     // private
48987     onRender : function(ct, position){
48988         if(this.el){ // from markup
48989             this.el = Roo.get(this.el);
48990         }else {  // generate
48991             var cfg = this.getAutoCreate();
48992             this.el = ct.createChild(cfg, position);
48993         }
48994         if(this.style){
48995             this.el.applyStyles(this.style);
48996         }
48997         if(this.labelAlign){
48998             this.el.addClass('x-form-label-'+this.labelAlign);
48999         }
49000         if(this.hideLabels){
49001             this.labelStyle = "display:none";
49002             this.elementStyle = "padding-left:0;";
49003         }else{
49004             if(typeof this.labelWidth == 'number'){
49005                 this.labelStyle = "width:"+this.labelWidth+"px;";
49006                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
49007             }
49008             if(this.labelAlign == 'top'){
49009                 this.labelStyle = "width:auto;";
49010                 this.elementStyle = "padding-left:0;";
49011             }
49012         }
49013         var stack = this.stack;
49014         var slen = stack.length;
49015         if(slen > 0){
49016             if(!this.fieldTpl){
49017                 var t = new Roo.Template(
49018                     '<div class="x-form-item {5}">',
49019                         '<label for="{0}" style="{2}">{1}{4}</label>',
49020                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49021                         '</div>',
49022                     '</div><div class="x-form-clear-left"></div>'
49023                 );
49024                 t.disableFormats = true;
49025                 t.compile();
49026                 Roo.form.Layout.prototype.fieldTpl = t;
49027             }
49028             for(var i = 0; i < slen; i++) {
49029                 if(stack[i].isFormField){
49030                     this.renderField(stack[i]);
49031                 }else{
49032                     this.renderComponent(stack[i]);
49033                 }
49034             }
49035         }
49036         if(this.clear){
49037             this.el.createChild({cls:'x-form-clear'});
49038         }
49039     },
49040
49041     // private
49042     renderField : function(f){
49043         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
49044                f.id, //0
49045                f.fieldLabel, //1
49046                f.labelStyle||this.labelStyle||'', //2
49047                this.elementStyle||'', //3
49048                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
49049                f.itemCls||this.itemCls||''  //5
49050        ], true).getPrevSibling());
49051     },
49052
49053     // private
49054     renderComponent : function(c){
49055         c.render(c.isLayout ? this.el : this.el.createChild());    
49056     },
49057     /**
49058      * Adds a object form elements (using the xtype property as the factory method.)
49059      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49060      * @param {Object} config 
49061      */
49062     addxtype : function(o)
49063     {
49064         // create the lement.
49065         o.form = this.form;
49066         var fe = Roo.factory(o, Roo.form);
49067         this.form.allItems.push(fe);
49068         this.stack.push(fe);
49069         
49070         if (fe.isFormField) {
49071             this.form.items.add(fe);
49072         }
49073          
49074         return fe;
49075     }
49076 });
49077
49078 /**
49079  * @class Roo.form.Column
49080  * @extends Roo.form.Layout
49081  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49082  * @constructor
49083  * @param {Object} config Configuration options
49084  */
49085 Roo.form.Column = function(config){
49086     Roo.form.Column.superclass.constructor.call(this, config);
49087 };
49088
49089 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49090     /**
49091      * @cfg {Number/String} width
49092      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49093      */
49094     /**
49095      * @cfg {String/Object} autoCreate
49096      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49097      */
49098
49099     // private
49100     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49101
49102     // private
49103     onRender : function(ct, position){
49104         Roo.form.Column.superclass.onRender.call(this, ct, position);
49105         if(this.width){
49106             this.el.setWidth(this.width);
49107         }
49108     }
49109 });
49110
49111
49112 /**
49113  * @class Roo.form.Row
49114  * @extends Roo.form.Layout
49115  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49116  * @constructor
49117  * @param {Object} config Configuration options
49118  */
49119
49120  
49121 Roo.form.Row = function(config){
49122     Roo.form.Row.superclass.constructor.call(this, config);
49123 };
49124  
49125 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49126       /**
49127      * @cfg {Number/String} width
49128      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49129      */
49130     /**
49131      * @cfg {Number/String} height
49132      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49133      */
49134     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49135     
49136     padWidth : 20,
49137     // private
49138     onRender : function(ct, position){
49139         //console.log('row render');
49140         if(!this.rowTpl){
49141             var t = new Roo.Template(
49142                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49143                     '<label for="{0}" style="{2}">{1}{4}</label>',
49144                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49145                     '</div>',
49146                 '</div>'
49147             );
49148             t.disableFormats = true;
49149             t.compile();
49150             Roo.form.Layout.prototype.rowTpl = t;
49151         }
49152         this.fieldTpl = this.rowTpl;
49153         
49154         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49155         var labelWidth = 100;
49156         
49157         if ((this.labelAlign != 'top')) {
49158             if (typeof this.labelWidth == 'number') {
49159                 labelWidth = this.labelWidth
49160             }
49161             this.padWidth =  20 + labelWidth;
49162             
49163         }
49164         
49165         Roo.form.Column.superclass.onRender.call(this, ct, position);
49166         if(this.width){
49167             this.el.setWidth(this.width);
49168         }
49169         if(this.height){
49170             this.el.setHeight(this.height);
49171         }
49172     },
49173     
49174     // private
49175     renderField : function(f){
49176         f.fieldEl = this.fieldTpl.append(this.el, [
49177                f.id, f.fieldLabel,
49178                f.labelStyle||this.labelStyle||'',
49179                this.elementStyle||'',
49180                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49181                f.itemCls||this.itemCls||'',
49182                f.width ? f.width + this.padWidth : 160 + this.padWidth
49183        ],true);
49184     }
49185 });
49186  
49187
49188 /**
49189  * @class Roo.form.FieldSet
49190  * @extends Roo.form.Layout
49191  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49192  * @constructor
49193  * @param {Object} config Configuration options
49194  */
49195 Roo.form.FieldSet = function(config){
49196     Roo.form.FieldSet.superclass.constructor.call(this, config);
49197 };
49198
49199 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49200     /**
49201      * @cfg {String} legend
49202      * The text to display as the legend for the FieldSet (defaults to '')
49203      */
49204     /**
49205      * @cfg {String/Object} autoCreate
49206      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49207      */
49208
49209     // private
49210     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49211
49212     // private
49213     onRender : function(ct, position){
49214         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49215         if(this.legend){
49216             this.setLegend(this.legend);
49217         }
49218     },
49219
49220     // private
49221     setLegend : function(text){
49222         if(this.rendered){
49223             this.el.child('legend').update(text);
49224         }
49225     }
49226 });/*
49227  * Based on:
49228  * Ext JS Library 1.1.1
49229  * Copyright(c) 2006-2007, Ext JS, LLC.
49230  *
49231  * Originally Released Under LGPL - original licence link has changed is not relivant.
49232  *
49233  * Fork - LGPL
49234  * <script type="text/javascript">
49235  */
49236 /**
49237  * @class Roo.form.VTypes
49238  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49239  * @singleton
49240  */
49241 Roo.form.VTypes = function(){
49242     // closure these in so they are only created once.
49243     var alpha = /^[a-zA-Z_]+$/;
49244     var alphanum = /^[a-zA-Z0-9_]+$/;
49245     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49246     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49247
49248     // All these messages and functions are configurable
49249     return {
49250         /**
49251          * The function used to validate email addresses
49252          * @param {String} value The email address
49253          */
49254         'email' : function(v){
49255             return email.test(v);
49256         },
49257         /**
49258          * The error text to display when the email validation function returns false
49259          * @type String
49260          */
49261         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49262         /**
49263          * The keystroke filter mask to be applied on email input
49264          * @type RegExp
49265          */
49266         'emailMask' : /[a-z0-9_\.\-@]/i,
49267
49268         /**
49269          * The function used to validate URLs
49270          * @param {String} value The URL
49271          */
49272         'url' : function(v){
49273             return url.test(v);
49274         },
49275         /**
49276          * The error text to display when the url validation function returns false
49277          * @type String
49278          */
49279         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49280         
49281         /**
49282          * The function used to validate alpha values
49283          * @param {String} value The value
49284          */
49285         'alpha' : function(v){
49286             return alpha.test(v);
49287         },
49288         /**
49289          * The error text to display when the alpha validation function returns false
49290          * @type String
49291          */
49292         'alphaText' : 'This field should only contain letters and _',
49293         /**
49294          * The keystroke filter mask to be applied on alpha input
49295          * @type RegExp
49296          */
49297         'alphaMask' : /[a-z_]/i,
49298
49299         /**
49300          * The function used to validate alphanumeric values
49301          * @param {String} value The value
49302          */
49303         'alphanum' : function(v){
49304             return alphanum.test(v);
49305         },
49306         /**
49307          * The error text to display when the alphanumeric validation function returns false
49308          * @type String
49309          */
49310         'alphanumText' : 'This field should only contain letters, numbers and _',
49311         /**
49312          * The keystroke filter mask to be applied on alphanumeric input
49313          * @type RegExp
49314          */
49315         'alphanumMask' : /[a-z0-9_]/i
49316     };
49317 }();//<script type="text/javascript">
49318
49319 /**
49320  * @class Roo.form.FCKeditor
49321  * @extends Roo.form.TextArea
49322  * Wrapper around the FCKEditor http://www.fckeditor.net
49323  * @constructor
49324  * Creates a new FCKeditor
49325  * @param {Object} config Configuration options
49326  */
49327 Roo.form.FCKeditor = function(config){
49328     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49329     this.addEvents({
49330          /**
49331          * @event editorinit
49332          * Fired when the editor is initialized - you can add extra handlers here..
49333          * @param {FCKeditor} this
49334          * @param {Object} the FCK object.
49335          */
49336         editorinit : true
49337     });
49338     
49339     
49340 };
49341 Roo.form.FCKeditor.editors = { };
49342 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49343 {
49344     //defaultAutoCreate : {
49345     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49346     //},
49347     // private
49348     /**
49349      * @cfg {Object} fck options - see fck manual for details.
49350      */
49351     fckconfig : false,
49352     
49353     /**
49354      * @cfg {Object} fck toolbar set (Basic or Default)
49355      */
49356     toolbarSet : 'Basic',
49357     /**
49358      * @cfg {Object} fck BasePath
49359      */ 
49360     basePath : '/fckeditor/',
49361     
49362     
49363     frame : false,
49364     
49365     value : '',
49366     
49367    
49368     onRender : function(ct, position)
49369     {
49370         if(!this.el){
49371             this.defaultAutoCreate = {
49372                 tag: "textarea",
49373                 style:"width:300px;height:60px;",
49374                 autocomplete: "new-password"
49375             };
49376         }
49377         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49378         /*
49379         if(this.grow){
49380             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49381             if(this.preventScrollbars){
49382                 this.el.setStyle("overflow", "hidden");
49383             }
49384             this.el.setHeight(this.growMin);
49385         }
49386         */
49387         //console.log('onrender' + this.getId() );
49388         Roo.form.FCKeditor.editors[this.getId()] = this;
49389          
49390
49391         this.replaceTextarea() ;
49392         
49393     },
49394     
49395     getEditor : function() {
49396         return this.fckEditor;
49397     },
49398     /**
49399      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49400      * @param {Mixed} value The value to set
49401      */
49402     
49403     
49404     setValue : function(value)
49405     {
49406         //console.log('setValue: ' + value);
49407         
49408         if(typeof(value) == 'undefined') { // not sure why this is happending...
49409             return;
49410         }
49411         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49412         
49413         //if(!this.el || !this.getEditor()) {
49414         //    this.value = value;
49415             //this.setValue.defer(100,this,[value]);    
49416         //    return;
49417         //} 
49418         
49419         if(!this.getEditor()) {
49420             return;
49421         }
49422         
49423         this.getEditor().SetData(value);
49424         
49425         //
49426
49427     },
49428
49429     /**
49430      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49431      * @return {Mixed} value The field value
49432      */
49433     getValue : function()
49434     {
49435         
49436         if (this.frame && this.frame.dom.style.display == 'none') {
49437             return Roo.form.FCKeditor.superclass.getValue.call(this);
49438         }
49439         
49440         if(!this.el || !this.getEditor()) {
49441            
49442            // this.getValue.defer(100,this); 
49443             return this.value;
49444         }
49445        
49446         
49447         var value=this.getEditor().GetData();
49448         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49449         return Roo.form.FCKeditor.superclass.getValue.call(this);
49450         
49451
49452     },
49453
49454     /**
49455      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49456      * @return {Mixed} value The field value
49457      */
49458     getRawValue : function()
49459     {
49460         if (this.frame && this.frame.dom.style.display == 'none') {
49461             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49462         }
49463         
49464         if(!this.el || !this.getEditor()) {
49465             //this.getRawValue.defer(100,this); 
49466             return this.value;
49467             return;
49468         }
49469         
49470         
49471         
49472         var value=this.getEditor().GetData();
49473         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49474         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49475          
49476     },
49477     
49478     setSize : function(w,h) {
49479         
49480         
49481         
49482         //if (this.frame && this.frame.dom.style.display == 'none') {
49483         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49484         //    return;
49485         //}
49486         //if(!this.el || !this.getEditor()) {
49487         //    this.setSize.defer(100,this, [w,h]); 
49488         //    return;
49489         //}
49490         
49491         
49492         
49493         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49494         
49495         this.frame.dom.setAttribute('width', w);
49496         this.frame.dom.setAttribute('height', h);
49497         this.frame.setSize(w,h);
49498         
49499     },
49500     
49501     toggleSourceEdit : function(value) {
49502         
49503       
49504          
49505         this.el.dom.style.display = value ? '' : 'none';
49506         this.frame.dom.style.display = value ?  'none' : '';
49507         
49508     },
49509     
49510     
49511     focus: function(tag)
49512     {
49513         if (this.frame.dom.style.display == 'none') {
49514             return Roo.form.FCKeditor.superclass.focus.call(this);
49515         }
49516         if(!this.el || !this.getEditor()) {
49517             this.focus.defer(100,this, [tag]); 
49518             return;
49519         }
49520         
49521         
49522         
49523         
49524         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49525         this.getEditor().Focus();
49526         if (tgs.length) {
49527             if (!this.getEditor().Selection.GetSelection()) {
49528                 this.focus.defer(100,this, [tag]); 
49529                 return;
49530             }
49531             
49532             
49533             var r = this.getEditor().EditorDocument.createRange();
49534             r.setStart(tgs[0],0);
49535             r.setEnd(tgs[0],0);
49536             this.getEditor().Selection.GetSelection().removeAllRanges();
49537             this.getEditor().Selection.GetSelection().addRange(r);
49538             this.getEditor().Focus();
49539         }
49540         
49541     },
49542     
49543     
49544     
49545     replaceTextarea : function()
49546     {
49547         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49548             return ;
49549         }
49550         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49551         //{
49552             // We must check the elements firstly using the Id and then the name.
49553         var oTextarea = document.getElementById( this.getId() );
49554         
49555         var colElementsByName = document.getElementsByName( this.getId() ) ;
49556          
49557         oTextarea.style.display = 'none' ;
49558
49559         if ( oTextarea.tabIndex ) {            
49560             this.TabIndex = oTextarea.tabIndex ;
49561         }
49562         
49563         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49564         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49565         this.frame = Roo.get(this.getId() + '___Frame')
49566     },
49567     
49568     _getConfigHtml : function()
49569     {
49570         var sConfig = '' ;
49571
49572         for ( var o in this.fckconfig ) {
49573             sConfig += sConfig.length > 0  ? '&amp;' : '';
49574             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49575         }
49576
49577         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49578     },
49579     
49580     
49581     _getIFrameHtml : function()
49582     {
49583         var sFile = 'fckeditor.html' ;
49584         /* no idea what this is about..
49585         try
49586         {
49587             if ( (/fcksource=true/i).test( window.top.location.search ) )
49588                 sFile = 'fckeditor.original.html' ;
49589         }
49590         catch (e) { 
49591         */
49592
49593         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49594         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49595         
49596         
49597         var html = '<iframe id="' + this.getId() +
49598             '___Frame" src="' + sLink +
49599             '" width="' + this.width +
49600             '" height="' + this.height + '"' +
49601             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49602             ' frameborder="0" scrolling="no"></iframe>' ;
49603
49604         return html ;
49605     },
49606     
49607     _insertHtmlBefore : function( html, element )
49608     {
49609         if ( element.insertAdjacentHTML )       {
49610             // IE
49611             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49612         } else { // Gecko
49613             var oRange = document.createRange() ;
49614             oRange.setStartBefore( element ) ;
49615             var oFragment = oRange.createContextualFragment( html );
49616             element.parentNode.insertBefore( oFragment, element ) ;
49617         }
49618     }
49619     
49620     
49621   
49622     
49623     
49624     
49625     
49626
49627 });
49628
49629 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49630
49631 function FCKeditor_OnComplete(editorInstance){
49632     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49633     f.fckEditor = editorInstance;
49634     //console.log("loaded");
49635     f.fireEvent('editorinit', f, editorInstance);
49636
49637   
49638
49639  
49640
49641
49642
49643
49644
49645
49646
49647
49648
49649
49650
49651
49652
49653
49654
49655 //<script type="text/javascript">
49656 /**
49657  * @class Roo.form.GridField
49658  * @extends Roo.form.Field
49659  * Embed a grid (or editable grid into a form)
49660  * STATUS ALPHA
49661  * 
49662  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49663  * it needs 
49664  * xgrid.store = Roo.data.Store
49665  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49666  * xgrid.store.reader = Roo.data.JsonReader 
49667  * 
49668  * 
49669  * @constructor
49670  * Creates a new GridField
49671  * @param {Object} config Configuration options
49672  */
49673 Roo.form.GridField = function(config){
49674     Roo.form.GridField.superclass.constructor.call(this, config);
49675      
49676 };
49677
49678 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49679     /**
49680      * @cfg {Number} width  - used to restrict width of grid..
49681      */
49682     width : 100,
49683     /**
49684      * @cfg {Number} height - used to restrict height of grid..
49685      */
49686     height : 50,
49687      /**
49688      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49689          * 
49690          *}
49691      */
49692     xgrid : false, 
49693     /**
49694      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49695      * {tag: "input", type: "checkbox", autocomplete: "off"})
49696      */
49697    // defaultAutoCreate : { tag: 'div' },
49698     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49699     /**
49700      * @cfg {String} addTitle Text to include for adding a title.
49701      */
49702     addTitle : false,
49703     //
49704     onResize : function(){
49705         Roo.form.Field.superclass.onResize.apply(this, arguments);
49706     },
49707
49708     initEvents : function(){
49709         // Roo.form.Checkbox.superclass.initEvents.call(this);
49710         // has no events...
49711        
49712     },
49713
49714
49715     getResizeEl : function(){
49716         return this.wrap;
49717     },
49718
49719     getPositionEl : function(){
49720         return this.wrap;
49721     },
49722
49723     // private
49724     onRender : function(ct, position){
49725         
49726         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49727         var style = this.style;
49728         delete this.style;
49729         
49730         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49731         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49732         this.viewEl = this.wrap.createChild({ tag: 'div' });
49733         if (style) {
49734             this.viewEl.applyStyles(style);
49735         }
49736         if (this.width) {
49737             this.viewEl.setWidth(this.width);
49738         }
49739         if (this.height) {
49740             this.viewEl.setHeight(this.height);
49741         }
49742         //if(this.inputValue !== undefined){
49743         //this.setValue(this.value);
49744         
49745         
49746         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49747         
49748         
49749         this.grid.render();
49750         this.grid.getDataSource().on('remove', this.refreshValue, this);
49751         this.grid.getDataSource().on('update', this.refreshValue, this);
49752         this.grid.on('afteredit', this.refreshValue, this);
49753  
49754     },
49755      
49756     
49757     /**
49758      * Sets the value of the item. 
49759      * @param {String} either an object  or a string..
49760      */
49761     setValue : function(v){
49762         //this.value = v;
49763         v = v || []; // empty set..
49764         // this does not seem smart - it really only affects memoryproxy grids..
49765         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49766             var ds = this.grid.getDataSource();
49767             // assumes a json reader..
49768             var data = {}
49769             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49770             ds.loadData( data);
49771         }
49772         // clear selection so it does not get stale.
49773         if (this.grid.sm) { 
49774             this.grid.sm.clearSelections();
49775         }
49776         
49777         Roo.form.GridField.superclass.setValue.call(this, v);
49778         this.refreshValue();
49779         // should load data in the grid really....
49780     },
49781     
49782     // private
49783     refreshValue: function() {
49784          var val = [];
49785         this.grid.getDataSource().each(function(r) {
49786             val.push(r.data);
49787         });
49788         this.el.dom.value = Roo.encode(val);
49789     }
49790     
49791      
49792     
49793     
49794 });/*
49795  * Based on:
49796  * Ext JS Library 1.1.1
49797  * Copyright(c) 2006-2007, Ext JS, LLC.
49798  *
49799  * Originally Released Under LGPL - original licence link has changed is not relivant.
49800  *
49801  * Fork - LGPL
49802  * <script type="text/javascript">
49803  */
49804 /**
49805  * @class Roo.form.DisplayField
49806  * @extends Roo.form.Field
49807  * A generic Field to display non-editable data.
49808  * @cfg {Boolean} closable (true|false) default false
49809  * @constructor
49810  * Creates a new Display Field item.
49811  * @param {Object} config Configuration options
49812  */
49813 Roo.form.DisplayField = function(config){
49814     Roo.form.DisplayField.superclass.constructor.call(this, config);
49815     
49816     this.addEvents({
49817         /**
49818          * @event close
49819          * Fires after the click the close btn
49820              * @param {Roo.form.DisplayField} this
49821              */
49822         close : true
49823     });
49824 };
49825
49826 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49827     inputType:      'hidden',
49828     allowBlank:     true,
49829     readOnly:         true,
49830     
49831  
49832     /**
49833      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49834      */
49835     focusClass : undefined,
49836     /**
49837      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49838      */
49839     fieldClass: 'x-form-field',
49840     
49841      /**
49842      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49843      */
49844     valueRenderer: undefined,
49845     
49846     width: 100,
49847     /**
49848      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49849      * {tag: "input", type: "checkbox", autocomplete: "off"})
49850      */
49851      
49852  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49853  
49854     closable : false,
49855     
49856     onResize : function(){
49857         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49858         
49859     },
49860
49861     initEvents : function(){
49862         // Roo.form.Checkbox.superclass.initEvents.call(this);
49863         // has no events...
49864         
49865         if(this.closable){
49866             this.closeEl.on('click', this.onClose, this);
49867         }
49868        
49869     },
49870
49871
49872     getResizeEl : function(){
49873         return this.wrap;
49874     },
49875
49876     getPositionEl : function(){
49877         return this.wrap;
49878     },
49879
49880     // private
49881     onRender : function(ct, position){
49882         
49883         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49884         //if(this.inputValue !== undefined){
49885         this.wrap = this.el.wrap();
49886         
49887         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49888         
49889         if(this.closable){
49890             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49891         }
49892         
49893         if (this.bodyStyle) {
49894             this.viewEl.applyStyles(this.bodyStyle);
49895         }
49896         //this.viewEl.setStyle('padding', '2px');
49897         
49898         this.setValue(this.value);
49899         
49900     },
49901 /*
49902     // private
49903     initValue : Roo.emptyFn,
49904
49905   */
49906
49907         // private
49908     onClick : function(){
49909         
49910     },
49911
49912     /**
49913      * Sets the checked state of the checkbox.
49914      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49915      */
49916     setValue : function(v){
49917         this.value = v;
49918         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49919         // this might be called before we have a dom element..
49920         if (!this.viewEl) {
49921             return;
49922         }
49923         this.viewEl.dom.innerHTML = html;
49924         Roo.form.DisplayField.superclass.setValue.call(this, v);
49925
49926     },
49927     
49928     onClose : function(e)
49929     {
49930         e.preventDefault();
49931         
49932         this.fireEvent('close', this);
49933     }
49934 });/*
49935  * 
49936  * Licence- LGPL
49937  * 
49938  */
49939
49940 /**
49941  * @class Roo.form.DayPicker
49942  * @extends Roo.form.Field
49943  * A Day picker show [M] [T] [W] ....
49944  * @constructor
49945  * Creates a new Day Picker
49946  * @param {Object} config Configuration options
49947  */
49948 Roo.form.DayPicker= function(config){
49949     Roo.form.DayPicker.superclass.constructor.call(this, config);
49950      
49951 };
49952
49953 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49954     /**
49955      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49956      */
49957     focusClass : undefined,
49958     /**
49959      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49960      */
49961     fieldClass: "x-form-field",
49962    
49963     /**
49964      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49965      * {tag: "input", type: "checkbox", autocomplete: "off"})
49966      */
49967     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49968     
49969    
49970     actionMode : 'viewEl', 
49971     //
49972     // private
49973  
49974     inputType : 'hidden',
49975     
49976      
49977     inputElement: false, // real input element?
49978     basedOn: false, // ????
49979     
49980     isFormField: true, // not sure where this is needed!!!!
49981
49982     onResize : function(){
49983         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49984         if(!this.boxLabel){
49985             this.el.alignTo(this.wrap, 'c-c');
49986         }
49987     },
49988
49989     initEvents : function(){
49990         Roo.form.Checkbox.superclass.initEvents.call(this);
49991         this.el.on("click", this.onClick,  this);
49992         this.el.on("change", this.onClick,  this);
49993     },
49994
49995
49996     getResizeEl : function(){
49997         return this.wrap;
49998     },
49999
50000     getPositionEl : function(){
50001         return this.wrap;
50002     },
50003
50004     
50005     // private
50006     onRender : function(ct, position){
50007         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
50008        
50009         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
50010         
50011         var r1 = '<table><tr>';
50012         var r2 = '<tr class="x-form-daypick-icons">';
50013         for (var i=0; i < 7; i++) {
50014             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
50015             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
50016         }
50017         
50018         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
50019         viewEl.select('img').on('click', this.onClick, this);
50020         this.viewEl = viewEl;   
50021         
50022         
50023         // this will not work on Chrome!!!
50024         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
50025         this.el.on('propertychange', this.setFromHidden,  this);  //ie
50026         
50027         
50028           
50029
50030     },
50031
50032     // private
50033     initValue : Roo.emptyFn,
50034
50035     /**
50036      * Returns the checked state of the checkbox.
50037      * @return {Boolean} True if checked, else false
50038      */
50039     getValue : function(){
50040         return this.el.dom.value;
50041         
50042     },
50043
50044         // private
50045     onClick : function(e){ 
50046         //this.setChecked(!this.checked);
50047         Roo.get(e.target).toggleClass('x-menu-item-checked');
50048         this.refreshValue();
50049         //if(this.el.dom.checked != this.checked){
50050         //    this.setValue(this.el.dom.checked);
50051        // }
50052     },
50053     
50054     // private
50055     refreshValue : function()
50056     {
50057         var val = '';
50058         this.viewEl.select('img',true).each(function(e,i,n)  {
50059             val += e.is(".x-menu-item-checked") ? String(n) : '';
50060         });
50061         this.setValue(val, true);
50062     },
50063
50064     /**
50065      * Sets the checked state of the checkbox.
50066      * On is always based on a string comparison between inputValue and the param.
50067      * @param {Boolean/String} value - the value to set 
50068      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50069      */
50070     setValue : function(v,suppressEvent){
50071         if (!this.el.dom) {
50072             return;
50073         }
50074         var old = this.el.dom.value ;
50075         this.el.dom.value = v;
50076         if (suppressEvent) {
50077             return ;
50078         }
50079          
50080         // update display..
50081         this.viewEl.select('img',true).each(function(e,i,n)  {
50082             
50083             var on = e.is(".x-menu-item-checked");
50084             var newv = v.indexOf(String(n)) > -1;
50085             if (on != newv) {
50086                 e.toggleClass('x-menu-item-checked');
50087             }
50088             
50089         });
50090         
50091         
50092         this.fireEvent('change', this, v, old);
50093         
50094         
50095     },
50096    
50097     // handle setting of hidden value by some other method!!?!?
50098     setFromHidden: function()
50099     {
50100         if(!this.el){
50101             return;
50102         }
50103         //console.log("SET FROM HIDDEN");
50104         //alert('setFrom hidden');
50105         this.setValue(this.el.dom.value);
50106     },
50107     
50108     onDestroy : function()
50109     {
50110         if(this.viewEl){
50111             Roo.get(this.viewEl).remove();
50112         }
50113          
50114         Roo.form.DayPicker.superclass.onDestroy.call(this);
50115     }
50116
50117 });/*
50118  * RooJS Library 1.1.1
50119  * Copyright(c) 2008-2011  Alan Knowles
50120  *
50121  * License - LGPL
50122  */
50123  
50124
50125 /**
50126  * @class Roo.form.ComboCheck
50127  * @extends Roo.form.ComboBox
50128  * A combobox for multiple select items.
50129  *
50130  * FIXME - could do with a reset button..
50131  * 
50132  * @constructor
50133  * Create a new ComboCheck
50134  * @param {Object} config Configuration options
50135  */
50136 Roo.form.ComboCheck = function(config){
50137     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50138     // should verify some data...
50139     // like
50140     // hiddenName = required..
50141     // displayField = required
50142     // valudField == required
50143     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50144     var _t = this;
50145     Roo.each(req, function(e) {
50146         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50147             throw "Roo.form.ComboCheck : missing value for: " + e;
50148         }
50149     });
50150     
50151     
50152 };
50153
50154 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50155      
50156      
50157     editable : false,
50158      
50159     selectedClass: 'x-menu-item-checked', 
50160     
50161     // private
50162     onRender : function(ct, position){
50163         var _t = this;
50164         
50165         
50166         
50167         if(!this.tpl){
50168             var cls = 'x-combo-list';
50169
50170             
50171             this.tpl =  new Roo.Template({
50172                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50173                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50174                    '<span>{' + this.displayField + '}</span>' +
50175                     '</div>' 
50176                 
50177             });
50178         }
50179  
50180         
50181         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50182         this.view.singleSelect = false;
50183         this.view.multiSelect = true;
50184         this.view.toggleSelect = true;
50185         this.pageTb.add(new Roo.Toolbar.Fill(), {
50186             
50187             text: 'Done',
50188             handler: function()
50189             {
50190                 _t.collapse();
50191             }
50192         });
50193     },
50194     
50195     onViewOver : function(e, t){
50196         // do nothing...
50197         return;
50198         
50199     },
50200     
50201     onViewClick : function(doFocus,index){
50202         return;
50203         
50204     },
50205     select: function () {
50206         //Roo.log("SELECT CALLED");
50207     },
50208      
50209     selectByValue : function(xv, scrollIntoView){
50210         var ar = this.getValueArray();
50211         var sels = [];
50212         
50213         Roo.each(ar, function(v) {
50214             if(v === undefined || v === null){
50215                 return;
50216             }
50217             var r = this.findRecord(this.valueField, v);
50218             if(r){
50219                 sels.push(this.store.indexOf(r))
50220                 
50221             }
50222         },this);
50223         this.view.select(sels);
50224         return false;
50225     },
50226     
50227     
50228     
50229     onSelect : function(record, index){
50230        // Roo.log("onselect Called");
50231        // this is only called by the clear button now..
50232         this.view.clearSelections();
50233         this.setValue('[]');
50234         if (this.value != this.valueBefore) {
50235             this.fireEvent('change', this, this.value, this.valueBefore);
50236             this.valueBefore = this.value;
50237         }
50238     },
50239     getValueArray : function()
50240     {
50241         var ar = [] ;
50242         
50243         try {
50244             //Roo.log(this.value);
50245             if (typeof(this.value) == 'undefined') {
50246                 return [];
50247             }
50248             var ar = Roo.decode(this.value);
50249             return  ar instanceof Array ? ar : []; //?? valid?
50250             
50251         } catch(e) {
50252             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50253             return [];
50254         }
50255          
50256     },
50257     expand : function ()
50258     {
50259         
50260         Roo.form.ComboCheck.superclass.expand.call(this);
50261         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50262         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50263         
50264
50265     },
50266     
50267     collapse : function(){
50268         Roo.form.ComboCheck.superclass.collapse.call(this);
50269         var sl = this.view.getSelectedIndexes();
50270         var st = this.store;
50271         var nv = [];
50272         var tv = [];
50273         var r;
50274         Roo.each(sl, function(i) {
50275             r = st.getAt(i);
50276             nv.push(r.get(this.valueField));
50277         },this);
50278         this.setValue(Roo.encode(nv));
50279         if (this.value != this.valueBefore) {
50280
50281             this.fireEvent('change', this, this.value, this.valueBefore);
50282             this.valueBefore = this.value;
50283         }
50284         
50285     },
50286     
50287     setValue : function(v){
50288         // Roo.log(v);
50289         this.value = v;
50290         
50291         var vals = this.getValueArray();
50292         var tv = [];
50293         Roo.each(vals, function(k) {
50294             var r = this.findRecord(this.valueField, k);
50295             if(r){
50296                 tv.push(r.data[this.displayField]);
50297             }else if(this.valueNotFoundText !== undefined){
50298                 tv.push( this.valueNotFoundText );
50299             }
50300         },this);
50301        // Roo.log(tv);
50302         
50303         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50304         this.hiddenField.value = v;
50305         this.value = v;
50306     }
50307     
50308 });/*
50309  * Based on:
50310  * Ext JS Library 1.1.1
50311  * Copyright(c) 2006-2007, Ext JS, LLC.
50312  *
50313  * Originally Released Under LGPL - original licence link has changed is not relivant.
50314  *
50315  * Fork - LGPL
50316  * <script type="text/javascript">
50317  */
50318  
50319 /**
50320  * @class Roo.form.Signature
50321  * @extends Roo.form.Field
50322  * Signature field.  
50323  * @constructor
50324  * 
50325  * @param {Object} config Configuration options
50326  */
50327
50328 Roo.form.Signature = function(config){
50329     Roo.form.Signature.superclass.constructor.call(this, config);
50330     
50331     this.addEvents({// not in used??
50332          /**
50333          * @event confirm
50334          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50335              * @param {Roo.form.Signature} combo This combo box
50336              */
50337         'confirm' : true,
50338         /**
50339          * @event reset
50340          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50341              * @param {Roo.form.ComboBox} combo This combo box
50342              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50343              */
50344         'reset' : true
50345     });
50346 };
50347
50348 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50349     /**
50350      * @cfg {Object} labels Label to use when rendering a form.
50351      * defaults to 
50352      * labels : { 
50353      *      clear : "Clear",
50354      *      confirm : "Confirm"
50355      *  }
50356      */
50357     labels : { 
50358         clear : "Clear",
50359         confirm : "Confirm"
50360     },
50361     /**
50362      * @cfg {Number} width The signature panel width (defaults to 300)
50363      */
50364     width: 300,
50365     /**
50366      * @cfg {Number} height The signature panel height (defaults to 100)
50367      */
50368     height : 100,
50369     /**
50370      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50371      */
50372     allowBlank : false,
50373     
50374     //private
50375     // {Object} signPanel The signature SVG panel element (defaults to {})
50376     signPanel : {},
50377     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50378     isMouseDown : false,
50379     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50380     isConfirmed : false,
50381     // {String} signatureTmp SVG mapping string (defaults to empty string)
50382     signatureTmp : '',
50383     
50384     
50385     defaultAutoCreate : { // modified by initCompnoent..
50386         tag: "input",
50387         type:"hidden"
50388     },
50389
50390     // private
50391     onRender : function(ct, position){
50392         
50393         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50394         
50395         this.wrap = this.el.wrap({
50396             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50397         });
50398         
50399         this.createToolbar(this);
50400         this.signPanel = this.wrap.createChild({
50401                 tag: 'div',
50402                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50403             }, this.el
50404         );
50405             
50406         this.svgID = Roo.id();
50407         this.svgEl = this.signPanel.createChild({
50408               xmlns : 'http://www.w3.org/2000/svg',
50409               tag : 'svg',
50410               id : this.svgID + "-svg",
50411               width: this.width,
50412               height: this.height,
50413               viewBox: '0 0 '+this.width+' '+this.height,
50414               cn : [
50415                 {
50416                     tag: "rect",
50417                     id: this.svgID + "-svg-r",
50418                     width: this.width,
50419                     height: this.height,
50420                     fill: "#ffa"
50421                 },
50422                 {
50423                     tag: "line",
50424                     id: this.svgID + "-svg-l",
50425                     x1: "0", // start
50426                     y1: (this.height*0.8), // start set the line in 80% of height
50427                     x2: this.width, // end
50428                     y2: (this.height*0.8), // end set the line in 80% of height
50429                     'stroke': "#666",
50430                     'stroke-width': "1",
50431                     'stroke-dasharray': "3",
50432                     'shape-rendering': "crispEdges",
50433                     'pointer-events': "none"
50434                 },
50435                 {
50436                     tag: "path",
50437                     id: this.svgID + "-svg-p",
50438                     'stroke': "navy",
50439                     'stroke-width': "3",
50440                     'fill': "none",
50441                     'pointer-events': 'none'
50442                 }
50443               ]
50444         });
50445         this.createSVG();
50446         this.svgBox = this.svgEl.dom.getScreenCTM();
50447     },
50448     createSVG : function(){ 
50449         var svg = this.signPanel;
50450         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50451         var t = this;
50452
50453         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50454         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50455         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50456         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50457         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50458         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50459         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50460         
50461     },
50462     isTouchEvent : function(e){
50463         return e.type.match(/^touch/);
50464     },
50465     getCoords : function (e) {
50466         var pt    = this.svgEl.dom.createSVGPoint();
50467         pt.x = e.clientX; 
50468         pt.y = e.clientY;
50469         if (this.isTouchEvent(e)) {
50470             pt.x =  e.targetTouches[0].clientX;
50471             pt.y = e.targetTouches[0].clientY;
50472         }
50473         var a = this.svgEl.dom.getScreenCTM();
50474         var b = a.inverse();
50475         var mx = pt.matrixTransform(b);
50476         return mx.x + ',' + mx.y;
50477     },
50478     //mouse event headler 
50479     down : function (e) {
50480         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50481         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50482         
50483         this.isMouseDown = true;
50484         
50485         e.preventDefault();
50486     },
50487     move : function (e) {
50488         if (this.isMouseDown) {
50489             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50490             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50491         }
50492         
50493         e.preventDefault();
50494     },
50495     up : function (e) {
50496         this.isMouseDown = false;
50497         var sp = this.signatureTmp.split(' ');
50498         
50499         if(sp.length > 1){
50500             if(!sp[sp.length-2].match(/^L/)){
50501                 sp.pop();
50502                 sp.pop();
50503                 sp.push("");
50504                 this.signatureTmp = sp.join(" ");
50505             }
50506         }
50507         if(this.getValue() != this.signatureTmp){
50508             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50509             this.isConfirmed = false;
50510         }
50511         e.preventDefault();
50512     },
50513     
50514     /**
50515      * Protected method that will not generally be called directly. It
50516      * is called when the editor creates its toolbar. Override this method if you need to
50517      * add custom toolbar buttons.
50518      * @param {HtmlEditor} editor
50519      */
50520     createToolbar : function(editor){
50521          function btn(id, toggle, handler){
50522             var xid = fid + '-'+ id ;
50523             return {
50524                 id : xid,
50525                 cmd : id,
50526                 cls : 'x-btn-icon x-edit-'+id,
50527                 enableToggle:toggle !== false,
50528                 scope: editor, // was editor...
50529                 handler:handler||editor.relayBtnCmd,
50530                 clickEvent:'mousedown',
50531                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50532                 tabIndex:-1
50533             };
50534         }
50535         
50536         
50537         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50538         this.tb = tb;
50539         this.tb.add(
50540            {
50541                 cls : ' x-signature-btn x-signature-'+id,
50542                 scope: editor, // was editor...
50543                 handler: this.reset,
50544                 clickEvent:'mousedown',
50545                 text: this.labels.clear
50546             },
50547             {
50548                  xtype : 'Fill',
50549                  xns: Roo.Toolbar
50550             }, 
50551             {
50552                 cls : '  x-signature-btn x-signature-'+id,
50553                 scope: editor, // was editor...
50554                 handler: this.confirmHandler,
50555                 clickEvent:'mousedown',
50556                 text: this.labels.confirm
50557             }
50558         );
50559     
50560     },
50561     //public
50562     /**
50563      * when user is clicked confirm then show this image.....
50564      * 
50565      * @return {String} Image Data URI
50566      */
50567     getImageDataURI : function(){
50568         var svg = this.svgEl.dom.parentNode.innerHTML;
50569         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50570         return src; 
50571     },
50572     /**
50573      * 
50574      * @return {Boolean} this.isConfirmed
50575      */
50576     getConfirmed : function(){
50577         return this.isConfirmed;
50578     },
50579     /**
50580      * 
50581      * @return {Number} this.width
50582      */
50583     getWidth : function(){
50584         return this.width;
50585     },
50586     /**
50587      * 
50588      * @return {Number} this.height
50589      */
50590     getHeight : function(){
50591         return this.height;
50592     },
50593     // private
50594     getSignature : function(){
50595         return this.signatureTmp;
50596     },
50597     // private
50598     reset : function(){
50599         this.signatureTmp = '';
50600         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50601         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50602         this.isConfirmed = false;
50603         Roo.form.Signature.superclass.reset.call(this);
50604     },
50605     setSignature : function(s){
50606         this.signatureTmp = s;
50607         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50608         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50609         this.setValue(s);
50610         this.isConfirmed = false;
50611         Roo.form.Signature.superclass.reset.call(this);
50612     }, 
50613     test : function(){
50614 //        Roo.log(this.signPanel.dom.contentWindow.up())
50615     },
50616     //private
50617     setConfirmed : function(){
50618         
50619         
50620         
50621 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50622     },
50623     // private
50624     confirmHandler : function(){
50625         if(!this.getSignature()){
50626             return;
50627         }
50628         
50629         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50630         this.setValue(this.getSignature());
50631         this.isConfirmed = true;
50632         
50633         this.fireEvent('confirm', this);
50634     },
50635     // private
50636     // Subclasses should provide the validation implementation by overriding this
50637     validateValue : function(value){
50638         if(this.allowBlank){
50639             return true;
50640         }
50641         
50642         if(this.isConfirmed){
50643             return true;
50644         }
50645         return false;
50646     }
50647 });/*
50648  * Based on:
50649  * Ext JS Library 1.1.1
50650  * Copyright(c) 2006-2007, Ext JS, LLC.
50651  *
50652  * Originally Released Under LGPL - original licence link has changed is not relivant.
50653  *
50654  * Fork - LGPL
50655  * <script type="text/javascript">
50656  */
50657  
50658
50659 /**
50660  * @class Roo.form.ComboBox
50661  * @extends Roo.form.TriggerField
50662  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50663  * @constructor
50664  * Create a new ComboBox.
50665  * @param {Object} config Configuration options
50666  */
50667 Roo.form.Select = function(config){
50668     Roo.form.Select.superclass.constructor.call(this, config);
50669      
50670 };
50671
50672 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50673     /**
50674      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50675      */
50676     /**
50677      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50678      * rendering into an Roo.Editor, defaults to false)
50679      */
50680     /**
50681      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50682      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50683      */
50684     /**
50685      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50686      */
50687     /**
50688      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50689      * the dropdown list (defaults to undefined, with no header element)
50690      */
50691
50692      /**
50693      * @cfg {String/Roo.Template} tpl The template to use to render the output
50694      */
50695      
50696     // private
50697     defaultAutoCreate : {tag: "select"  },
50698     /**
50699      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50700      */
50701     listWidth: undefined,
50702     /**
50703      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50704      * mode = 'remote' or 'text' if mode = 'local')
50705      */
50706     displayField: undefined,
50707     /**
50708      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50709      * mode = 'remote' or 'value' if mode = 'local'). 
50710      * Note: use of a valueField requires the user make a selection
50711      * in order for a value to be mapped.
50712      */
50713     valueField: undefined,
50714     
50715     
50716     /**
50717      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50718      * field's data value (defaults to the underlying DOM element's name)
50719      */
50720     hiddenName: undefined,
50721     /**
50722      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50723      */
50724     listClass: '',
50725     /**
50726      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50727      */
50728     selectedClass: 'x-combo-selected',
50729     /**
50730      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50731      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50732      * which displays a downward arrow icon).
50733      */
50734     triggerClass : 'x-form-arrow-trigger',
50735     /**
50736      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50737      */
50738     shadow:'sides',
50739     /**
50740      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50741      * anchor positions (defaults to 'tl-bl')
50742      */
50743     listAlign: 'tl-bl?',
50744     /**
50745      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50746      */
50747     maxHeight: 300,
50748     /**
50749      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50750      * query specified by the allQuery config option (defaults to 'query')
50751      */
50752     triggerAction: 'query',
50753     /**
50754      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50755      * (defaults to 4, does not apply if editable = false)
50756      */
50757     minChars : 4,
50758     /**
50759      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50760      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50761      */
50762     typeAhead: false,
50763     /**
50764      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50765      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50766      */
50767     queryDelay: 500,
50768     /**
50769      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50770      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50771      */
50772     pageSize: 0,
50773     /**
50774      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50775      * when editable = true (defaults to false)
50776      */
50777     selectOnFocus:false,
50778     /**
50779      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50780      */
50781     queryParam: 'query',
50782     /**
50783      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50784      * when mode = 'remote' (defaults to 'Loading...')
50785      */
50786     loadingText: 'Loading...',
50787     /**
50788      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50789      */
50790     resizable: false,
50791     /**
50792      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50793      */
50794     handleHeight : 8,
50795     /**
50796      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50797      * traditional select (defaults to true)
50798      */
50799     editable: true,
50800     /**
50801      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50802      */
50803     allQuery: '',
50804     /**
50805      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50806      */
50807     mode: 'remote',
50808     /**
50809      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50810      * listWidth has a higher value)
50811      */
50812     minListWidth : 70,
50813     /**
50814      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50815      * allow the user to set arbitrary text into the field (defaults to false)
50816      */
50817     forceSelection:false,
50818     /**
50819      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50820      * if typeAhead = true (defaults to 250)
50821      */
50822     typeAheadDelay : 250,
50823     /**
50824      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50825      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50826      */
50827     valueNotFoundText : undefined,
50828     
50829     /**
50830      * @cfg {String} defaultValue The value displayed after loading the store.
50831      */
50832     defaultValue: '',
50833     
50834     /**
50835      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50836      */
50837     blockFocus : false,
50838     
50839     /**
50840      * @cfg {Boolean} disableClear Disable showing of clear button.
50841      */
50842     disableClear : false,
50843     /**
50844      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50845      */
50846     alwaysQuery : false,
50847     
50848     //private
50849     addicon : false,
50850     editicon: false,
50851     
50852     // element that contains real text value.. (when hidden is used..)
50853      
50854     // private
50855     onRender : function(ct, position){
50856         Roo.form.Field.prototype.onRender.call(this, ct, position);
50857         
50858         if(this.store){
50859             this.store.on('beforeload', this.onBeforeLoad, this);
50860             this.store.on('load', this.onLoad, this);
50861             this.store.on('loadexception', this.onLoadException, this);
50862             this.store.load({});
50863         }
50864         
50865         
50866         
50867     },
50868
50869     // private
50870     initEvents : function(){
50871         //Roo.form.ComboBox.superclass.initEvents.call(this);
50872  
50873     },
50874
50875     onDestroy : function(){
50876        
50877         if(this.store){
50878             this.store.un('beforeload', this.onBeforeLoad, this);
50879             this.store.un('load', this.onLoad, this);
50880             this.store.un('loadexception', this.onLoadException, this);
50881         }
50882         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50883     },
50884
50885     // private
50886     fireKey : function(e){
50887         if(e.isNavKeyPress() && !this.list.isVisible()){
50888             this.fireEvent("specialkey", this, e);
50889         }
50890     },
50891
50892     // private
50893     onResize: function(w, h){
50894         
50895         return; 
50896     
50897         
50898     },
50899
50900     /**
50901      * Allow or prevent the user from directly editing the field text.  If false is passed,
50902      * the user will only be able to select from the items defined in the dropdown list.  This method
50903      * is the runtime equivalent of setting the 'editable' config option at config time.
50904      * @param {Boolean} value True to allow the user to directly edit the field text
50905      */
50906     setEditable : function(value){
50907          
50908     },
50909
50910     // private
50911     onBeforeLoad : function(){
50912         
50913         Roo.log("Select before load");
50914         return;
50915     
50916         this.innerList.update(this.loadingText ?
50917                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50918         //this.restrictHeight();
50919         this.selectedIndex = -1;
50920     },
50921
50922     // private
50923     onLoad : function(){
50924
50925     
50926         var dom = this.el.dom;
50927         dom.innerHTML = '';
50928          var od = dom.ownerDocument;
50929          
50930         if (this.emptyText) {
50931             var op = od.createElement('option');
50932             op.setAttribute('value', '');
50933             op.innerHTML = String.format('{0}', this.emptyText);
50934             dom.appendChild(op);
50935         }
50936         if(this.store.getCount() > 0){
50937            
50938             var vf = this.valueField;
50939             var df = this.displayField;
50940             this.store.data.each(function(r) {
50941                 // which colmsn to use... testing - cdoe / title..
50942                 var op = od.createElement('option');
50943                 op.setAttribute('value', r.data[vf]);
50944                 op.innerHTML = String.format('{0}', r.data[df]);
50945                 dom.appendChild(op);
50946             });
50947             if (typeof(this.defaultValue != 'undefined')) {
50948                 this.setValue(this.defaultValue);
50949             }
50950             
50951              
50952         }else{
50953             //this.onEmptyResults();
50954         }
50955         //this.el.focus();
50956     },
50957     // private
50958     onLoadException : function()
50959     {
50960         dom.innerHTML = '';
50961             
50962         Roo.log("Select on load exception");
50963         return;
50964     
50965         this.collapse();
50966         Roo.log(this.store.reader.jsonData);
50967         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50968             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50969         }
50970         
50971         
50972     },
50973     // private
50974     onTypeAhead : function(){
50975          
50976     },
50977
50978     // private
50979     onSelect : function(record, index){
50980         Roo.log('on select?');
50981         return;
50982         if(this.fireEvent('beforeselect', this, record, index) !== false){
50983             this.setFromData(index > -1 ? record.data : false);
50984             this.collapse();
50985             this.fireEvent('select', this, record, index);
50986         }
50987     },
50988
50989     /**
50990      * Returns the currently selected field value or empty string if no value is set.
50991      * @return {String} value The selected value
50992      */
50993     getValue : function(){
50994         var dom = this.el.dom;
50995         this.value = dom.options[dom.selectedIndex].value;
50996         return this.value;
50997         
50998     },
50999
51000     /**
51001      * Clears any text/value currently set in the field
51002      */
51003     clearValue : function(){
51004         this.value = '';
51005         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
51006         
51007     },
51008
51009     /**
51010      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
51011      * will be displayed in the field.  If the value does not match the data value of an existing item,
51012      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
51013      * Otherwise the field will be blank (although the value will still be set).
51014      * @param {String} value The value to match
51015      */
51016     setValue : function(v){
51017         var d = this.el.dom;
51018         for (var i =0; i < d.options.length;i++) {
51019             if (v == d.options[i].value) {
51020                 d.selectedIndex = i;
51021                 this.value = v;
51022                 return;
51023             }
51024         }
51025         this.clearValue();
51026     },
51027     /**
51028      * @property {Object} the last set data for the element
51029      */
51030     
51031     lastData : false,
51032     /**
51033      * Sets the value of the field based on a object which is related to the record format for the store.
51034      * @param {Object} value the value to set as. or false on reset?
51035      */
51036     setFromData : function(o){
51037         Roo.log('setfrom data?');
51038          
51039         
51040         
51041     },
51042     // private
51043     reset : function(){
51044         this.clearValue();
51045     },
51046     // private
51047     findRecord : function(prop, value){
51048         
51049         return false;
51050     
51051         var record;
51052         if(this.store.getCount() > 0){
51053             this.store.each(function(r){
51054                 if(r.data[prop] == value){
51055                     record = r;
51056                     return false;
51057                 }
51058                 return true;
51059             });
51060         }
51061         return record;
51062     },
51063     
51064     getName: function()
51065     {
51066         // returns hidden if it's set..
51067         if (!this.rendered) {return ''};
51068         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51069         
51070     },
51071      
51072
51073     
51074
51075     // private
51076     onEmptyResults : function(){
51077         Roo.log('empty results');
51078         //this.collapse();
51079     },
51080
51081     /**
51082      * Returns true if the dropdown list is expanded, else false.
51083      */
51084     isExpanded : function(){
51085         return false;
51086     },
51087
51088     /**
51089      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51090      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51091      * @param {String} value The data value of the item to select
51092      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51093      * selected item if it is not currently in view (defaults to true)
51094      * @return {Boolean} True if the value matched an item in the list, else false
51095      */
51096     selectByValue : function(v, scrollIntoView){
51097         Roo.log('select By Value');
51098         return false;
51099     
51100         if(v !== undefined && v !== null){
51101             var r = this.findRecord(this.valueField || this.displayField, v);
51102             if(r){
51103                 this.select(this.store.indexOf(r), scrollIntoView);
51104                 return true;
51105             }
51106         }
51107         return false;
51108     },
51109
51110     /**
51111      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51112      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51113      * @param {Number} index The zero-based index of the list item to select
51114      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51115      * selected item if it is not currently in view (defaults to true)
51116      */
51117     select : function(index, scrollIntoView){
51118         Roo.log('select ');
51119         return  ;
51120         
51121         this.selectedIndex = index;
51122         this.view.select(index);
51123         if(scrollIntoView !== false){
51124             var el = this.view.getNode(index);
51125             if(el){
51126                 this.innerList.scrollChildIntoView(el, false);
51127             }
51128         }
51129     },
51130
51131       
51132
51133     // private
51134     validateBlur : function(){
51135         
51136         return;
51137         
51138     },
51139
51140     // private
51141     initQuery : function(){
51142         this.doQuery(this.getRawValue());
51143     },
51144
51145     // private
51146     doForce : function(){
51147         if(this.el.dom.value.length > 0){
51148             this.el.dom.value =
51149                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51150              
51151         }
51152     },
51153
51154     /**
51155      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51156      * query allowing the query action to be canceled if needed.
51157      * @param {String} query The SQL query to execute
51158      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51159      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51160      * saved in the current store (defaults to false)
51161      */
51162     doQuery : function(q, forceAll){
51163         
51164         Roo.log('doQuery?');
51165         if(q === undefined || q === null){
51166             q = '';
51167         }
51168         var qe = {
51169             query: q,
51170             forceAll: forceAll,
51171             combo: this,
51172             cancel:false
51173         };
51174         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51175             return false;
51176         }
51177         q = qe.query;
51178         forceAll = qe.forceAll;
51179         if(forceAll === true || (q.length >= this.minChars)){
51180             if(this.lastQuery != q || this.alwaysQuery){
51181                 this.lastQuery = q;
51182                 if(this.mode == 'local'){
51183                     this.selectedIndex = -1;
51184                     if(forceAll){
51185                         this.store.clearFilter();
51186                     }else{
51187                         this.store.filter(this.displayField, q);
51188                     }
51189                     this.onLoad();
51190                 }else{
51191                     this.store.baseParams[this.queryParam] = q;
51192                     this.store.load({
51193                         params: this.getParams(q)
51194                     });
51195                     this.expand();
51196                 }
51197             }else{
51198                 this.selectedIndex = -1;
51199                 this.onLoad();   
51200             }
51201         }
51202     },
51203
51204     // private
51205     getParams : function(q){
51206         var p = {};
51207         //p[this.queryParam] = q;
51208         if(this.pageSize){
51209             p.start = 0;
51210             p.limit = this.pageSize;
51211         }
51212         return p;
51213     },
51214
51215     /**
51216      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51217      */
51218     collapse : function(){
51219         
51220     },
51221
51222     // private
51223     collapseIf : function(e){
51224         
51225     },
51226
51227     /**
51228      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51229      */
51230     expand : function(){
51231         
51232     } ,
51233
51234     // private
51235      
51236
51237     /** 
51238     * @cfg {Boolean} grow 
51239     * @hide 
51240     */
51241     /** 
51242     * @cfg {Number} growMin 
51243     * @hide 
51244     */
51245     /** 
51246     * @cfg {Number} growMax 
51247     * @hide 
51248     */
51249     /**
51250      * @hide
51251      * @method autoSize
51252      */
51253     
51254     setWidth : function()
51255     {
51256         
51257     },
51258     getResizeEl : function(){
51259         return this.el;
51260     }
51261 });//<script type="text/javasscript">
51262  
51263
51264 /**
51265  * @class Roo.DDView
51266  * A DnD enabled version of Roo.View.
51267  * @param {Element/String} container The Element in which to create the View.
51268  * @param {String} tpl The template string used to create the markup for each element of the View
51269  * @param {Object} config The configuration properties. These include all the config options of
51270  * {@link Roo.View} plus some specific to this class.<br>
51271  * <p>
51272  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51273  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51274  * <p>
51275  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51276 .x-view-drag-insert-above {
51277         border-top:1px dotted #3366cc;
51278 }
51279 .x-view-drag-insert-below {
51280         border-bottom:1px dotted #3366cc;
51281 }
51282 </code></pre>
51283  * 
51284  */
51285  
51286 Roo.DDView = function(container, tpl, config) {
51287     Roo.DDView.superclass.constructor.apply(this, arguments);
51288     this.getEl().setStyle("outline", "0px none");
51289     this.getEl().unselectable();
51290     if (this.dragGroup) {
51291                 this.setDraggable(this.dragGroup.split(","));
51292     }
51293     if (this.dropGroup) {
51294                 this.setDroppable(this.dropGroup.split(","));
51295     }
51296     if (this.deletable) {
51297         this.setDeletable();
51298     }
51299     this.isDirtyFlag = false;
51300         this.addEvents({
51301                 "drop" : true
51302         });
51303 };
51304
51305 Roo.extend(Roo.DDView, Roo.View, {
51306 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51307 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51308 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51309 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51310
51311         isFormField: true,
51312
51313         reset: Roo.emptyFn,
51314         
51315         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51316
51317         validate: function() {
51318                 return true;
51319         },
51320         
51321         destroy: function() {
51322                 this.purgeListeners();
51323                 this.getEl.removeAllListeners();
51324                 this.getEl().remove();
51325                 if (this.dragZone) {
51326                         if (this.dragZone.destroy) {
51327                                 this.dragZone.destroy();
51328                         }
51329                 }
51330                 if (this.dropZone) {
51331                         if (this.dropZone.destroy) {
51332                                 this.dropZone.destroy();
51333                         }
51334                 }
51335         },
51336
51337 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51338         getName: function() {
51339                 return this.name;
51340         },
51341
51342 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51343         setValue: function(v) {
51344                 if (!this.store) {
51345                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51346                 }
51347                 var data = {};
51348                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51349                 this.store.proxy = new Roo.data.MemoryProxy(data);
51350                 this.store.load();
51351         },
51352
51353 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51354         getValue: function() {
51355                 var result = '(';
51356                 this.store.each(function(rec) {
51357                         result += rec.id + ',';
51358                 });
51359                 return result.substr(0, result.length - 1) + ')';
51360         },
51361         
51362         getIds: function() {
51363                 var i = 0, result = new Array(this.store.getCount());
51364                 this.store.each(function(rec) {
51365                         result[i++] = rec.id;
51366                 });
51367                 return result;
51368         },
51369         
51370         isDirty: function() {
51371                 return this.isDirtyFlag;
51372         },
51373
51374 /**
51375  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51376  *      whole Element becomes the target, and this causes the drop gesture to append.
51377  */
51378     getTargetFromEvent : function(e) {
51379                 var target = e.getTarget();
51380                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51381                 target = target.parentNode;
51382                 }
51383                 if (!target) {
51384                         target = this.el.dom.lastChild || this.el.dom;
51385                 }
51386                 return target;
51387     },
51388
51389 /**
51390  *      Create the drag data which consists of an object which has the property "ddel" as
51391  *      the drag proxy element. 
51392  */
51393     getDragData : function(e) {
51394         var target = this.findItemFromChild(e.getTarget());
51395                 if(target) {
51396                         this.handleSelection(e);
51397                         var selNodes = this.getSelectedNodes();
51398             var dragData = {
51399                 source: this,
51400                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51401                 nodes: selNodes,
51402                 records: []
51403                         };
51404                         var selectedIndices = this.getSelectedIndexes();
51405                         for (var i = 0; i < selectedIndices.length; i++) {
51406                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51407                         }
51408                         if (selNodes.length == 1) {
51409                                 dragData.ddel = target.cloneNode(true); // the div element
51410                         } else {
51411                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51412                                 div.className = 'multi-proxy';
51413                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51414                                         div.appendChild(selNodes[i].cloneNode(true));
51415                                 }
51416                                 dragData.ddel = div;
51417                         }
51418             //console.log(dragData)
51419             //console.log(dragData.ddel.innerHTML)
51420                         return dragData;
51421                 }
51422         //console.log('nodragData')
51423                 return false;
51424     },
51425     
51426 /**     Specify to which ddGroup items in this DDView may be dragged. */
51427     setDraggable: function(ddGroup) {
51428         if (ddGroup instanceof Array) {
51429                 Roo.each(ddGroup, this.setDraggable, this);
51430                 return;
51431         }
51432         if (this.dragZone) {
51433                 this.dragZone.addToGroup(ddGroup);
51434         } else {
51435                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51436                                 containerScroll: true,
51437                                 ddGroup: ddGroup 
51438
51439                         });
51440 //                      Draggability implies selection. DragZone's mousedown selects the element.
51441                         if (!this.multiSelect) { this.singleSelect = true; }
51442
51443 //                      Wire the DragZone's handlers up to methods in *this*
51444                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51445                 }
51446     },
51447
51448 /**     Specify from which ddGroup this DDView accepts drops. */
51449     setDroppable: function(ddGroup) {
51450         if (ddGroup instanceof Array) {
51451                 Roo.each(ddGroup, this.setDroppable, this);
51452                 return;
51453         }
51454         if (this.dropZone) {
51455                 this.dropZone.addToGroup(ddGroup);
51456         } else {
51457                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51458                                 containerScroll: true,
51459                                 ddGroup: ddGroup
51460                         });
51461
51462 //                      Wire the DropZone's handlers up to methods in *this*
51463                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51464                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51465                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51466                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51467                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51468                 }
51469     },
51470
51471 /**     Decide whether to drop above or below a View node. */
51472     getDropPoint : function(e, n, dd){
51473         if (n == this.el.dom) { return "above"; }
51474                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51475                 var c = t + (b - t) / 2;
51476                 var y = Roo.lib.Event.getPageY(e);
51477                 if(y <= c) {
51478                         return "above";
51479                 }else{
51480                         return "below";
51481                 }
51482     },
51483
51484     onNodeEnter : function(n, dd, e, data){
51485                 return false;
51486     },
51487     
51488     onNodeOver : function(n, dd, e, data){
51489                 var pt = this.getDropPoint(e, n, dd);
51490                 // set the insert point style on the target node
51491                 var dragElClass = this.dropNotAllowed;
51492                 if (pt) {
51493                         var targetElClass;
51494                         if (pt == "above"){
51495                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51496                                 targetElClass = "x-view-drag-insert-above";
51497                         } else {
51498                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51499                                 targetElClass = "x-view-drag-insert-below";
51500                         }
51501                         if (this.lastInsertClass != targetElClass){
51502                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51503                                 this.lastInsertClass = targetElClass;
51504                         }
51505                 }
51506                 return dragElClass;
51507         },
51508
51509     onNodeOut : function(n, dd, e, data){
51510                 this.removeDropIndicators(n);
51511     },
51512
51513     onNodeDrop : function(n, dd, e, data){
51514         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51515                 return false;
51516         }
51517         var pt = this.getDropPoint(e, n, dd);
51518                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51519                 if (pt == "below") { insertAt++; }
51520                 for (var i = 0; i < data.records.length; i++) {
51521                         var r = data.records[i];
51522                         var dup = this.store.getById(r.id);
51523                         if (dup && (dd != this.dragZone)) {
51524                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51525                         } else {
51526                                 if (data.copy) {
51527                                         this.store.insert(insertAt++, r.copy());
51528                                 } else {
51529                                         data.source.isDirtyFlag = true;
51530                                         r.store.remove(r);
51531                                         this.store.insert(insertAt++, r);
51532                                 }
51533                                 this.isDirtyFlag = true;
51534                         }
51535                 }
51536                 this.dragZone.cachedTarget = null;
51537                 return true;
51538     },
51539
51540     removeDropIndicators : function(n){
51541                 if(n){
51542                         Roo.fly(n).removeClass([
51543                                 "x-view-drag-insert-above",
51544                                 "x-view-drag-insert-below"]);
51545                         this.lastInsertClass = "_noclass";
51546                 }
51547     },
51548
51549 /**
51550  *      Utility method. Add a delete option to the DDView's context menu.
51551  *      @param {String} imageUrl The URL of the "delete" icon image.
51552  */
51553         setDeletable: function(imageUrl) {
51554                 if (!this.singleSelect && !this.multiSelect) {
51555                         this.singleSelect = true;
51556                 }
51557                 var c = this.getContextMenu();
51558                 this.contextMenu.on("itemclick", function(item) {
51559                         switch (item.id) {
51560                                 case "delete":
51561                                         this.remove(this.getSelectedIndexes());
51562                                         break;
51563                         }
51564                 }, this);
51565                 this.contextMenu.add({
51566                         icon: imageUrl,
51567                         id: "delete",
51568                         text: 'Delete'
51569                 });
51570         },
51571         
51572 /**     Return the context menu for this DDView. */
51573         getContextMenu: function() {
51574                 if (!this.contextMenu) {
51575 //                      Create the View's context menu
51576                         this.contextMenu = new Roo.menu.Menu({
51577                                 id: this.id + "-contextmenu"
51578                         });
51579                         this.el.on("contextmenu", this.showContextMenu, this);
51580                 }
51581                 return this.contextMenu;
51582         },
51583         
51584         disableContextMenu: function() {
51585                 if (this.contextMenu) {
51586                         this.el.un("contextmenu", this.showContextMenu, this);
51587                 }
51588         },
51589
51590         showContextMenu: function(e, item) {
51591         item = this.findItemFromChild(e.getTarget());
51592                 if (item) {
51593                         e.stopEvent();
51594                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51595                         this.contextMenu.showAt(e.getXY());
51596             }
51597     },
51598
51599 /**
51600  *      Remove {@link Roo.data.Record}s at the specified indices.
51601  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51602  */
51603     remove: function(selectedIndices) {
51604                 selectedIndices = [].concat(selectedIndices);
51605                 for (var i = 0; i < selectedIndices.length; i++) {
51606                         var rec = this.store.getAt(selectedIndices[i]);
51607                         this.store.remove(rec);
51608                 }
51609     },
51610
51611 /**
51612  *      Double click fires the event, but also, if this is draggable, and there is only one other
51613  *      related DropZone, it transfers the selected node.
51614  */
51615     onDblClick : function(e){
51616         var item = this.findItemFromChild(e.getTarget());
51617         if(item){
51618             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51619                 return false;
51620             }
51621             if (this.dragGroup) {
51622                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51623                     while (targets.indexOf(this.dropZone) > -1) {
51624                             targets.remove(this.dropZone);
51625                                 }
51626                     if (targets.length == 1) {
51627                                         this.dragZone.cachedTarget = null;
51628                         var el = Roo.get(targets[0].getEl());
51629                         var box = el.getBox(true);
51630                         targets[0].onNodeDrop(el.dom, {
51631                                 target: el.dom,
51632                                 xy: [box.x, box.y + box.height - 1]
51633                         }, null, this.getDragData(e));
51634                     }
51635                 }
51636         }
51637     },
51638     
51639     handleSelection: function(e) {
51640                 this.dragZone.cachedTarget = null;
51641         var item = this.findItemFromChild(e.getTarget());
51642         if (!item) {
51643                 this.clearSelections(true);
51644                 return;
51645         }
51646                 if (item && (this.multiSelect || this.singleSelect)){
51647                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51648                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51649                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51650                                 this.unselect(item);
51651                         } else {
51652                                 this.select(item, this.multiSelect && e.ctrlKey);
51653                                 this.lastSelection = item;
51654                         }
51655                 }
51656     },
51657
51658     onItemClick : function(item, index, e){
51659                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51660                         return false;
51661                 }
51662                 return true;
51663     },
51664
51665     unselect : function(nodeInfo, suppressEvent){
51666                 var node = this.getNode(nodeInfo);
51667                 if(node && this.isSelected(node)){
51668                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51669                                 Roo.fly(node).removeClass(this.selectedClass);
51670                                 this.selections.remove(node);
51671                                 if(!suppressEvent){
51672                                         this.fireEvent("selectionchange", this, this.selections);
51673                                 }
51674                         }
51675                 }
51676     }
51677 });
51678 /*
51679  * Based on:
51680  * Ext JS Library 1.1.1
51681  * Copyright(c) 2006-2007, Ext JS, LLC.
51682  *
51683  * Originally Released Under LGPL - original licence link has changed is not relivant.
51684  *
51685  * Fork - LGPL
51686  * <script type="text/javascript">
51687  */
51688  
51689 /**
51690  * @class Roo.LayoutManager
51691  * @extends Roo.util.Observable
51692  * Base class for layout managers.
51693  */
51694 Roo.LayoutManager = function(container, config){
51695     Roo.LayoutManager.superclass.constructor.call(this);
51696     this.el = Roo.get(container);
51697     // ie scrollbar fix
51698     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51699         document.body.scroll = "no";
51700     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51701         this.el.position('relative');
51702     }
51703     this.id = this.el.id;
51704     this.el.addClass("x-layout-container");
51705     /** false to disable window resize monitoring @type Boolean */
51706     this.monitorWindowResize = true;
51707     this.regions = {};
51708     this.addEvents({
51709         /**
51710          * @event layout
51711          * Fires when a layout is performed. 
51712          * @param {Roo.LayoutManager} this
51713          */
51714         "layout" : true,
51715         /**
51716          * @event regionresized
51717          * Fires when the user resizes a region. 
51718          * @param {Roo.LayoutRegion} region The resized region
51719          * @param {Number} newSize The new size (width for east/west, height for north/south)
51720          */
51721         "regionresized" : true,
51722         /**
51723          * @event regioncollapsed
51724          * Fires when a region is collapsed. 
51725          * @param {Roo.LayoutRegion} region The collapsed region
51726          */
51727         "regioncollapsed" : true,
51728         /**
51729          * @event regionexpanded
51730          * Fires when a region is expanded.  
51731          * @param {Roo.LayoutRegion} region The expanded region
51732          */
51733         "regionexpanded" : true
51734     });
51735     this.updating = false;
51736     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51737 };
51738
51739 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51740     /**
51741      * Returns true if this layout is currently being updated
51742      * @return {Boolean}
51743      */
51744     isUpdating : function(){
51745         return this.updating; 
51746     },
51747     
51748     /**
51749      * Suspend the LayoutManager from doing auto-layouts while
51750      * making multiple add or remove calls
51751      */
51752     beginUpdate : function(){
51753         this.updating = true;    
51754     },
51755     
51756     /**
51757      * Restore auto-layouts and optionally disable the manager from performing a layout
51758      * @param {Boolean} noLayout true to disable a layout update 
51759      */
51760     endUpdate : function(noLayout){
51761         this.updating = false;
51762         if(!noLayout){
51763             this.layout();
51764         }    
51765     },
51766     
51767     layout: function(){
51768         
51769     },
51770     
51771     onRegionResized : function(region, newSize){
51772         this.fireEvent("regionresized", region, newSize);
51773         this.layout();
51774     },
51775     
51776     onRegionCollapsed : function(region){
51777         this.fireEvent("regioncollapsed", region);
51778     },
51779     
51780     onRegionExpanded : function(region){
51781         this.fireEvent("regionexpanded", region);
51782     },
51783         
51784     /**
51785      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51786      * performs box-model adjustments.
51787      * @return {Object} The size as an object {width: (the width), height: (the height)}
51788      */
51789     getViewSize : function(){
51790         var size;
51791         if(this.el.dom != document.body){
51792             size = this.el.getSize();
51793         }else{
51794             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51795         }
51796         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51797         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51798         return size;
51799     },
51800     
51801     /**
51802      * Returns the Element this layout is bound to.
51803      * @return {Roo.Element}
51804      */
51805     getEl : function(){
51806         return this.el;
51807     },
51808     
51809     /**
51810      * Returns the specified region.
51811      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51812      * @return {Roo.LayoutRegion}
51813      */
51814     getRegion : function(target){
51815         return this.regions[target.toLowerCase()];
51816     },
51817     
51818     onWindowResize : function(){
51819         if(this.monitorWindowResize){
51820             this.layout();
51821         }
51822     }
51823 });/*
51824  * Based on:
51825  * Ext JS Library 1.1.1
51826  * Copyright(c) 2006-2007, Ext JS, LLC.
51827  *
51828  * Originally Released Under LGPL - original licence link has changed is not relivant.
51829  *
51830  * Fork - LGPL
51831  * <script type="text/javascript">
51832  */
51833 /**
51834  * @class Roo.BorderLayout
51835  * @extends Roo.LayoutManager
51836  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51837  * please see: <br><br>
51838  * <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>
51839  * <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>
51840  * Example:
51841  <pre><code>
51842  var layout = new Roo.BorderLayout(document.body, {
51843     north: {
51844         initialSize: 25,
51845         titlebar: false
51846     },
51847     west: {
51848         split:true,
51849         initialSize: 200,
51850         minSize: 175,
51851         maxSize: 400,
51852         titlebar: true,
51853         collapsible: true
51854     },
51855     east: {
51856         split:true,
51857         initialSize: 202,
51858         minSize: 175,
51859         maxSize: 400,
51860         titlebar: true,
51861         collapsible: true
51862     },
51863     south: {
51864         split:true,
51865         initialSize: 100,
51866         minSize: 100,
51867         maxSize: 200,
51868         titlebar: true,
51869         collapsible: true
51870     },
51871     center: {
51872         titlebar: true,
51873         autoScroll:true,
51874         resizeTabs: true,
51875         minTabWidth: 50,
51876         preferredTabWidth: 150
51877     }
51878 });
51879
51880 // shorthand
51881 var CP = Roo.ContentPanel;
51882
51883 layout.beginUpdate();
51884 layout.add("north", new CP("north", "North"));
51885 layout.add("south", new CP("south", {title: "South", closable: true}));
51886 layout.add("west", new CP("west", {title: "West"}));
51887 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51888 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51889 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51890 layout.getRegion("center").showPanel("center1");
51891 layout.endUpdate();
51892 </code></pre>
51893
51894 <b>The container the layout is rendered into can be either the body element or any other element.
51895 If it is not the body element, the container needs to either be an absolute positioned element,
51896 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51897 the container size if it is not the body element.</b>
51898
51899 * @constructor
51900 * Create a new BorderLayout
51901 * @param {String/HTMLElement/Element} container The container this layout is bound to
51902 * @param {Object} config Configuration options
51903  */
51904 Roo.BorderLayout = function(container, config){
51905     config = config || {};
51906     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51907     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51908     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51909         var target = this.factory.validRegions[i];
51910         if(config[target]){
51911             this.addRegion(target, config[target]);
51912         }
51913     }
51914 };
51915
51916 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51917     /**
51918      * Creates and adds a new region if it doesn't already exist.
51919      * @param {String} target The target region key (north, south, east, west or center).
51920      * @param {Object} config The regions config object
51921      * @return {BorderLayoutRegion} The new region
51922      */
51923     addRegion : function(target, config){
51924         if(!this.regions[target]){
51925             var r = this.factory.create(target, this, config);
51926             this.bindRegion(target, r);
51927         }
51928         return this.regions[target];
51929     },
51930
51931     // private (kinda)
51932     bindRegion : function(name, r){
51933         this.regions[name] = r;
51934         r.on("visibilitychange", this.layout, this);
51935         r.on("paneladded", this.layout, this);
51936         r.on("panelremoved", this.layout, this);
51937         r.on("invalidated", this.layout, this);
51938         r.on("resized", this.onRegionResized, this);
51939         r.on("collapsed", this.onRegionCollapsed, this);
51940         r.on("expanded", this.onRegionExpanded, this);
51941     },
51942
51943     /**
51944      * Performs a layout update.
51945      */
51946     layout : function(){
51947         if(this.updating) {
51948             return;
51949         }
51950         var size = this.getViewSize();
51951         var w = size.width;
51952         var h = size.height;
51953         var centerW = w;
51954         var centerH = h;
51955         var centerY = 0;
51956         var centerX = 0;
51957         //var x = 0, y = 0;
51958
51959         var rs = this.regions;
51960         var north = rs["north"];
51961         var south = rs["south"]; 
51962         var west = rs["west"];
51963         var east = rs["east"];
51964         var center = rs["center"];
51965         //if(this.hideOnLayout){ // not supported anymore
51966             //c.el.setStyle("display", "none");
51967         //}
51968         if(north && north.isVisible()){
51969             var b = north.getBox();
51970             var m = north.getMargins();
51971             b.width = w - (m.left+m.right);
51972             b.x = m.left;
51973             b.y = m.top;
51974             centerY = b.height + b.y + m.bottom;
51975             centerH -= centerY;
51976             north.updateBox(this.safeBox(b));
51977         }
51978         if(south && south.isVisible()){
51979             var b = south.getBox();
51980             var m = south.getMargins();
51981             b.width = w - (m.left+m.right);
51982             b.x = m.left;
51983             var totalHeight = (b.height + m.top + m.bottom);
51984             b.y = h - totalHeight + m.top;
51985             centerH -= totalHeight;
51986             south.updateBox(this.safeBox(b));
51987         }
51988         if(west && west.isVisible()){
51989             var b = west.getBox();
51990             var m = west.getMargins();
51991             b.height = centerH - (m.top+m.bottom);
51992             b.x = m.left;
51993             b.y = centerY + m.top;
51994             var totalWidth = (b.width + m.left + m.right);
51995             centerX += totalWidth;
51996             centerW -= totalWidth;
51997             west.updateBox(this.safeBox(b));
51998         }
51999         if(east && east.isVisible()){
52000             var b = east.getBox();
52001             var m = east.getMargins();
52002             b.height = centerH - (m.top+m.bottom);
52003             var totalWidth = (b.width + m.left + m.right);
52004             b.x = w - totalWidth + m.left;
52005             b.y = centerY + m.top;
52006             centerW -= totalWidth;
52007             east.updateBox(this.safeBox(b));
52008         }
52009         if(center){
52010             var m = center.getMargins();
52011             var centerBox = {
52012                 x: centerX + m.left,
52013                 y: centerY + m.top,
52014                 width: centerW - (m.left+m.right),
52015                 height: centerH - (m.top+m.bottom)
52016             };
52017             //if(this.hideOnLayout){
52018                 //center.el.setStyle("display", "block");
52019             //}
52020             center.updateBox(this.safeBox(centerBox));
52021         }
52022         this.el.repaint();
52023         this.fireEvent("layout", this);
52024     },
52025
52026     // private
52027     safeBox : function(box){
52028         box.width = Math.max(0, box.width);
52029         box.height = Math.max(0, box.height);
52030         return box;
52031     },
52032
52033     /**
52034      * Adds a ContentPanel (or subclass) to this layout.
52035      * @param {String} target The target region key (north, south, east, west or center).
52036      * @param {Roo.ContentPanel} panel The panel to add
52037      * @return {Roo.ContentPanel} The added panel
52038      */
52039     add : function(target, panel){
52040          
52041         target = target.toLowerCase();
52042         return this.regions[target].add(panel);
52043     },
52044
52045     /**
52046      * Remove a ContentPanel (or subclass) to this layout.
52047      * @param {String} target The target region key (north, south, east, west or center).
52048      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
52049      * @return {Roo.ContentPanel} The removed panel
52050      */
52051     remove : function(target, panel){
52052         target = target.toLowerCase();
52053         return this.regions[target].remove(panel);
52054     },
52055
52056     /**
52057      * Searches all regions for a panel with the specified id
52058      * @param {String} panelId
52059      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52060      */
52061     findPanel : function(panelId){
52062         var rs = this.regions;
52063         for(var target in rs){
52064             if(typeof rs[target] != "function"){
52065                 var p = rs[target].getPanel(panelId);
52066                 if(p){
52067                     return p;
52068                 }
52069             }
52070         }
52071         return null;
52072     },
52073
52074     /**
52075      * Searches all regions for a panel with the specified id and activates (shows) it.
52076      * @param {String/ContentPanel} panelId The panels id or the panel itself
52077      * @return {Roo.ContentPanel} The shown panel or null
52078      */
52079     showPanel : function(panelId) {
52080       var rs = this.regions;
52081       for(var target in rs){
52082          var r = rs[target];
52083          if(typeof r != "function"){
52084             if(r.hasPanel(panelId)){
52085                return r.showPanel(panelId);
52086             }
52087          }
52088       }
52089       return null;
52090    },
52091
52092    /**
52093      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52094      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52095      */
52096     restoreState : function(provider){
52097         if(!provider){
52098             provider = Roo.state.Manager;
52099         }
52100         var sm = new Roo.LayoutStateManager();
52101         sm.init(this, provider);
52102     },
52103
52104     /**
52105      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52106      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52107      * a valid ContentPanel config object.  Example:
52108      * <pre><code>
52109 // Create the main layout
52110 var layout = new Roo.BorderLayout('main-ct', {
52111     west: {
52112         split:true,
52113         minSize: 175,
52114         titlebar: true
52115     },
52116     center: {
52117         title:'Components'
52118     }
52119 }, 'main-ct');
52120
52121 // Create and add multiple ContentPanels at once via configs
52122 layout.batchAdd({
52123    west: {
52124        id: 'source-files',
52125        autoCreate:true,
52126        title:'Ext Source Files',
52127        autoScroll:true,
52128        fitToFrame:true
52129    },
52130    center : {
52131        el: cview,
52132        autoScroll:true,
52133        fitToFrame:true,
52134        toolbar: tb,
52135        resizeEl:'cbody'
52136    }
52137 });
52138 </code></pre>
52139      * @param {Object} regions An object containing ContentPanel configs by region name
52140      */
52141     batchAdd : function(regions){
52142         this.beginUpdate();
52143         for(var rname in regions){
52144             var lr = this.regions[rname];
52145             if(lr){
52146                 this.addTypedPanels(lr, regions[rname]);
52147             }
52148         }
52149         this.endUpdate();
52150     },
52151
52152     // private
52153     addTypedPanels : function(lr, ps){
52154         if(typeof ps == 'string'){
52155             lr.add(new Roo.ContentPanel(ps));
52156         }
52157         else if(ps instanceof Array){
52158             for(var i =0, len = ps.length; i < len; i++){
52159                 this.addTypedPanels(lr, ps[i]);
52160             }
52161         }
52162         else if(!ps.events){ // raw config?
52163             var el = ps.el;
52164             delete ps.el; // prevent conflict
52165             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52166         }
52167         else {  // panel object assumed!
52168             lr.add(ps);
52169         }
52170     },
52171     /**
52172      * Adds a xtype elements to the layout.
52173      * <pre><code>
52174
52175 layout.addxtype({
52176        xtype : 'ContentPanel',
52177        region: 'west',
52178        items: [ .... ]
52179    }
52180 );
52181
52182 layout.addxtype({
52183         xtype : 'NestedLayoutPanel',
52184         region: 'west',
52185         layout: {
52186            center: { },
52187            west: { }   
52188         },
52189         items : [ ... list of content panels or nested layout panels.. ]
52190    }
52191 );
52192 </code></pre>
52193      * @param {Object} cfg Xtype definition of item to add.
52194      */
52195     addxtype : function(cfg)
52196     {
52197         // basically accepts a pannel...
52198         // can accept a layout region..!?!?
52199         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52200         
52201         if (!cfg.xtype.match(/Panel$/)) {
52202             return false;
52203         }
52204         var ret = false;
52205         
52206         if (typeof(cfg.region) == 'undefined') {
52207             Roo.log("Failed to add Panel, region was not set");
52208             Roo.log(cfg);
52209             return false;
52210         }
52211         var region = cfg.region;
52212         delete cfg.region;
52213         
52214           
52215         var xitems = [];
52216         if (cfg.items) {
52217             xitems = cfg.items;
52218             delete cfg.items;
52219         }
52220         var nb = false;
52221         
52222         switch(cfg.xtype) 
52223         {
52224             case 'ContentPanel':  // ContentPanel (el, cfg)
52225             case 'ScrollPanel':  // ContentPanel (el, cfg)
52226             case 'ViewPanel': 
52227                 if(cfg.autoCreate) {
52228                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52229                 } else {
52230                     var el = this.el.createChild();
52231                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52232                 }
52233                 
52234                 this.add(region, ret);
52235                 break;
52236             
52237             
52238             case 'TreePanel': // our new panel!
52239                 cfg.el = this.el.createChild();
52240                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52241                 this.add(region, ret);
52242                 break;
52243             
52244             case 'NestedLayoutPanel': 
52245                 // create a new Layout (which is  a Border Layout...
52246                 var el = this.el.createChild();
52247                 var clayout = cfg.layout;
52248                 delete cfg.layout;
52249                 clayout.items   = clayout.items  || [];
52250                 // replace this exitems with the clayout ones..
52251                 xitems = clayout.items;
52252                  
52253                 
52254                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52255                     cfg.background = false;
52256                 }
52257                 var layout = new Roo.BorderLayout(el, clayout);
52258                 
52259                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52260                 //console.log('adding nested layout panel '  + cfg.toSource());
52261                 this.add(region, ret);
52262                 nb = {}; /// find first...
52263                 break;
52264                 
52265             case 'GridPanel': 
52266             
52267                 // needs grid and region
52268                 
52269                 //var el = this.getRegion(region).el.createChild();
52270                 var el = this.el.createChild();
52271                 // create the grid first...
52272                 
52273                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52274                 delete cfg.grid;
52275                 if (region == 'center' && this.active ) {
52276                     cfg.background = false;
52277                 }
52278                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52279                 
52280                 this.add(region, ret);
52281                 if (cfg.background) {
52282                     ret.on('activate', function(gp) {
52283                         if (!gp.grid.rendered) {
52284                             gp.grid.render();
52285                         }
52286                     });
52287                 } else {
52288                     grid.render();
52289                 }
52290                 break;
52291            
52292            
52293            
52294                 
52295                 
52296                 
52297             default:
52298                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52299                     
52300                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52301                     this.add(region, ret);
52302                 } else {
52303                 
52304                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52305                     return null;
52306                 }
52307                 
52308              // GridPanel (grid, cfg)
52309             
52310         }
52311         this.beginUpdate();
52312         // add children..
52313         var region = '';
52314         var abn = {};
52315         Roo.each(xitems, function(i)  {
52316             region = nb && i.region ? i.region : false;
52317             
52318             var add = ret.addxtype(i);
52319            
52320             if (region) {
52321                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52322                 if (!i.background) {
52323                     abn[region] = nb[region] ;
52324                 }
52325             }
52326             
52327         });
52328         this.endUpdate();
52329
52330         // make the last non-background panel active..
52331         //if (nb) { Roo.log(abn); }
52332         if (nb) {
52333             
52334             for(var r in abn) {
52335                 region = this.getRegion(r);
52336                 if (region) {
52337                     // tried using nb[r], but it does not work..
52338                      
52339                     region.showPanel(abn[r]);
52340                    
52341                 }
52342             }
52343         }
52344         return ret;
52345         
52346     }
52347 });
52348
52349 /**
52350  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52351  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52352  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52353  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52354  * <pre><code>
52355 // shorthand
52356 var CP = Roo.ContentPanel;
52357
52358 var layout = Roo.BorderLayout.create({
52359     north: {
52360         initialSize: 25,
52361         titlebar: false,
52362         panels: [new CP("north", "North")]
52363     },
52364     west: {
52365         split:true,
52366         initialSize: 200,
52367         minSize: 175,
52368         maxSize: 400,
52369         titlebar: true,
52370         collapsible: true,
52371         panels: [new CP("west", {title: "West"})]
52372     },
52373     east: {
52374         split:true,
52375         initialSize: 202,
52376         minSize: 175,
52377         maxSize: 400,
52378         titlebar: true,
52379         collapsible: true,
52380         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52381     },
52382     south: {
52383         split:true,
52384         initialSize: 100,
52385         minSize: 100,
52386         maxSize: 200,
52387         titlebar: true,
52388         collapsible: true,
52389         panels: [new CP("south", {title: "South", closable: true})]
52390     },
52391     center: {
52392         titlebar: true,
52393         autoScroll:true,
52394         resizeTabs: true,
52395         minTabWidth: 50,
52396         preferredTabWidth: 150,
52397         panels: [
52398             new CP("center1", {title: "Close Me", closable: true}),
52399             new CP("center2", {title: "Center Panel", closable: false})
52400         ]
52401     }
52402 }, document.body);
52403
52404 layout.getRegion("center").showPanel("center1");
52405 </code></pre>
52406  * @param config
52407  * @param targetEl
52408  */
52409 Roo.BorderLayout.create = function(config, targetEl){
52410     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52411     layout.beginUpdate();
52412     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52413     for(var j = 0, jlen = regions.length; j < jlen; j++){
52414         var lr = regions[j];
52415         if(layout.regions[lr] && config[lr].panels){
52416             var r = layout.regions[lr];
52417             var ps = config[lr].panels;
52418             layout.addTypedPanels(r, ps);
52419         }
52420     }
52421     layout.endUpdate();
52422     return layout;
52423 };
52424
52425 // private
52426 Roo.BorderLayout.RegionFactory = {
52427     // private
52428     validRegions : ["north","south","east","west","center"],
52429
52430     // private
52431     create : function(target, mgr, config){
52432         target = target.toLowerCase();
52433         if(config.lightweight || config.basic){
52434             return new Roo.BasicLayoutRegion(mgr, config, target);
52435         }
52436         switch(target){
52437             case "north":
52438                 return new Roo.NorthLayoutRegion(mgr, config);
52439             case "south":
52440                 return new Roo.SouthLayoutRegion(mgr, config);
52441             case "east":
52442                 return new Roo.EastLayoutRegion(mgr, config);
52443             case "west":
52444                 return new Roo.WestLayoutRegion(mgr, config);
52445             case "center":
52446                 return new Roo.CenterLayoutRegion(mgr, config);
52447         }
52448         throw 'Layout region "'+target+'" not supported.';
52449     }
52450 };/*
52451  * Based on:
52452  * Ext JS Library 1.1.1
52453  * Copyright(c) 2006-2007, Ext JS, LLC.
52454  *
52455  * Originally Released Under LGPL - original licence link has changed is not relivant.
52456  *
52457  * Fork - LGPL
52458  * <script type="text/javascript">
52459  */
52460  
52461 /**
52462  * @class Roo.BasicLayoutRegion
52463  * @extends Roo.util.Observable
52464  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52465  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52466  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52467  */
52468 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52469     this.mgr = mgr;
52470     this.position  = pos;
52471     this.events = {
52472         /**
52473          * @scope Roo.BasicLayoutRegion
52474          */
52475         
52476         /**
52477          * @event beforeremove
52478          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52479          * @param {Roo.LayoutRegion} this
52480          * @param {Roo.ContentPanel} panel The panel
52481          * @param {Object} e The cancel event object
52482          */
52483         "beforeremove" : true,
52484         /**
52485          * @event invalidated
52486          * Fires when the layout for this region is changed.
52487          * @param {Roo.LayoutRegion} this
52488          */
52489         "invalidated" : true,
52490         /**
52491          * @event visibilitychange
52492          * Fires when this region is shown or hidden 
52493          * @param {Roo.LayoutRegion} this
52494          * @param {Boolean} visibility true or false
52495          */
52496         "visibilitychange" : true,
52497         /**
52498          * @event paneladded
52499          * Fires when a panel is added. 
52500          * @param {Roo.LayoutRegion} this
52501          * @param {Roo.ContentPanel} panel The panel
52502          */
52503         "paneladded" : true,
52504         /**
52505          * @event panelremoved
52506          * Fires when a panel is removed. 
52507          * @param {Roo.LayoutRegion} this
52508          * @param {Roo.ContentPanel} panel The panel
52509          */
52510         "panelremoved" : true,
52511         /**
52512          * @event beforecollapse
52513          * Fires when this region before collapse.
52514          * @param {Roo.LayoutRegion} this
52515          */
52516         "beforecollapse" : true,
52517         /**
52518          * @event collapsed
52519          * Fires when this region is collapsed.
52520          * @param {Roo.LayoutRegion} this
52521          */
52522         "collapsed" : true,
52523         /**
52524          * @event expanded
52525          * Fires when this region is expanded.
52526          * @param {Roo.LayoutRegion} this
52527          */
52528         "expanded" : true,
52529         /**
52530          * @event slideshow
52531          * Fires when this region is slid into view.
52532          * @param {Roo.LayoutRegion} this
52533          */
52534         "slideshow" : true,
52535         /**
52536          * @event slidehide
52537          * Fires when this region slides out of view. 
52538          * @param {Roo.LayoutRegion} this
52539          */
52540         "slidehide" : true,
52541         /**
52542          * @event panelactivated
52543          * Fires when a panel is activated. 
52544          * @param {Roo.LayoutRegion} this
52545          * @param {Roo.ContentPanel} panel The activated panel
52546          */
52547         "panelactivated" : true,
52548         /**
52549          * @event resized
52550          * Fires when the user resizes this region. 
52551          * @param {Roo.LayoutRegion} this
52552          * @param {Number} newSize The new size (width for east/west, height for north/south)
52553          */
52554         "resized" : true
52555     };
52556     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52557     this.panels = new Roo.util.MixedCollection();
52558     this.panels.getKey = this.getPanelId.createDelegate(this);
52559     this.box = null;
52560     this.activePanel = null;
52561     // ensure listeners are added...
52562     
52563     if (config.listeners || config.events) {
52564         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52565             listeners : config.listeners || {},
52566             events : config.events || {}
52567         });
52568     }
52569     
52570     if(skipConfig !== true){
52571         this.applyConfig(config);
52572     }
52573 };
52574
52575 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52576     getPanelId : function(p){
52577         return p.getId();
52578     },
52579     
52580     applyConfig : function(config){
52581         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52582         this.config = config;
52583         
52584     },
52585     
52586     /**
52587      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52588      * the width, for horizontal (north, south) the height.
52589      * @param {Number} newSize The new width or height
52590      */
52591     resizeTo : function(newSize){
52592         var el = this.el ? this.el :
52593                  (this.activePanel ? this.activePanel.getEl() : null);
52594         if(el){
52595             switch(this.position){
52596                 case "east":
52597                 case "west":
52598                     el.setWidth(newSize);
52599                     this.fireEvent("resized", this, newSize);
52600                 break;
52601                 case "north":
52602                 case "south":
52603                     el.setHeight(newSize);
52604                     this.fireEvent("resized", this, newSize);
52605                 break;                
52606             }
52607         }
52608     },
52609     
52610     getBox : function(){
52611         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52612     },
52613     
52614     getMargins : function(){
52615         return this.margins;
52616     },
52617     
52618     updateBox : function(box){
52619         this.box = box;
52620         var el = this.activePanel.getEl();
52621         el.dom.style.left = box.x + "px";
52622         el.dom.style.top = box.y + "px";
52623         this.activePanel.setSize(box.width, box.height);
52624     },
52625     
52626     /**
52627      * Returns the container element for this region.
52628      * @return {Roo.Element}
52629      */
52630     getEl : function(){
52631         return this.activePanel;
52632     },
52633     
52634     /**
52635      * Returns true if this region is currently visible.
52636      * @return {Boolean}
52637      */
52638     isVisible : function(){
52639         return this.activePanel ? true : false;
52640     },
52641     
52642     setActivePanel : function(panel){
52643         panel = this.getPanel(panel);
52644         if(this.activePanel && this.activePanel != panel){
52645             this.activePanel.setActiveState(false);
52646             this.activePanel.getEl().setLeftTop(-10000,-10000);
52647         }
52648         this.activePanel = panel;
52649         panel.setActiveState(true);
52650         if(this.box){
52651             panel.setSize(this.box.width, this.box.height);
52652         }
52653         this.fireEvent("panelactivated", this, panel);
52654         this.fireEvent("invalidated");
52655     },
52656     
52657     /**
52658      * Show the specified panel.
52659      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52660      * @return {Roo.ContentPanel} The shown panel or null
52661      */
52662     showPanel : function(panel){
52663         if(panel = this.getPanel(panel)){
52664             this.setActivePanel(panel);
52665         }
52666         return panel;
52667     },
52668     
52669     /**
52670      * Get the active panel for this region.
52671      * @return {Roo.ContentPanel} The active panel or null
52672      */
52673     getActivePanel : function(){
52674         return this.activePanel;
52675     },
52676     
52677     /**
52678      * Add the passed ContentPanel(s)
52679      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52680      * @return {Roo.ContentPanel} The panel added (if only one was added)
52681      */
52682     add : function(panel){
52683         if(arguments.length > 1){
52684             for(var i = 0, len = arguments.length; i < len; i++) {
52685                 this.add(arguments[i]);
52686             }
52687             return null;
52688         }
52689         if(this.hasPanel(panel)){
52690             this.showPanel(panel);
52691             return panel;
52692         }
52693         var el = panel.getEl();
52694         if(el.dom.parentNode != this.mgr.el.dom){
52695             this.mgr.el.dom.appendChild(el.dom);
52696         }
52697         if(panel.setRegion){
52698             panel.setRegion(this);
52699         }
52700         this.panels.add(panel);
52701         el.setStyle("position", "absolute");
52702         if(!panel.background){
52703             this.setActivePanel(panel);
52704             if(this.config.initialSize && this.panels.getCount()==1){
52705                 this.resizeTo(this.config.initialSize);
52706             }
52707         }
52708         this.fireEvent("paneladded", this, panel);
52709         return panel;
52710     },
52711     
52712     /**
52713      * Returns true if the panel is in this region.
52714      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52715      * @return {Boolean}
52716      */
52717     hasPanel : function(panel){
52718         if(typeof panel == "object"){ // must be panel obj
52719             panel = panel.getId();
52720         }
52721         return this.getPanel(panel) ? true : false;
52722     },
52723     
52724     /**
52725      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52726      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52727      * @param {Boolean} preservePanel Overrides the config preservePanel option
52728      * @return {Roo.ContentPanel} The panel that was removed
52729      */
52730     remove : function(panel, preservePanel){
52731         panel = this.getPanel(panel);
52732         if(!panel){
52733             return null;
52734         }
52735         var e = {};
52736         this.fireEvent("beforeremove", this, panel, e);
52737         if(e.cancel === true){
52738             return null;
52739         }
52740         var panelId = panel.getId();
52741         this.panels.removeKey(panelId);
52742         return panel;
52743     },
52744     
52745     /**
52746      * Returns the panel specified or null if it's not in this region.
52747      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52748      * @return {Roo.ContentPanel}
52749      */
52750     getPanel : function(id){
52751         if(typeof id == "object"){ // must be panel obj
52752             return id;
52753         }
52754         return this.panels.get(id);
52755     },
52756     
52757     /**
52758      * Returns this regions position (north/south/east/west/center).
52759      * @return {String} 
52760      */
52761     getPosition: function(){
52762         return this.position;    
52763     }
52764 });/*
52765  * Based on:
52766  * Ext JS Library 1.1.1
52767  * Copyright(c) 2006-2007, Ext JS, LLC.
52768  *
52769  * Originally Released Under LGPL - original licence link has changed is not relivant.
52770  *
52771  * Fork - LGPL
52772  * <script type="text/javascript">
52773  */
52774  
52775 /**
52776  * @class Roo.LayoutRegion
52777  * @extends Roo.BasicLayoutRegion
52778  * This class represents a region in a layout manager.
52779  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52780  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52781  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52782  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52783  * @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})
52784  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52785  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52786  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52787  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52788  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52789  * @cfg {String}    title           The title for the region (overrides panel titles)
52790  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52791  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52792  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52793  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52794  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52795  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52796  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52797  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52798  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52799  * @cfg {Boolean}   showPin         True to show a pin button
52800  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52801  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52802  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52803  * @cfg {Number}    width           For East/West panels
52804  * @cfg {Number}    height          For North/South panels
52805  * @cfg {Boolean}   split           To show the splitter
52806  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52807  */
52808 Roo.LayoutRegion = function(mgr, config, pos){
52809     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52810     var dh = Roo.DomHelper;
52811     /** This region's container element 
52812     * @type Roo.Element */
52813     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52814     /** This region's title element 
52815     * @type Roo.Element */
52816
52817     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52818         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52819         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52820     ]}, true);
52821     this.titleEl.enableDisplayMode();
52822     /** This region's title text element 
52823     * @type HTMLElement */
52824     this.titleTextEl = this.titleEl.dom.firstChild;
52825     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52826     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52827     this.closeBtn.enableDisplayMode();
52828     this.closeBtn.on("click", this.closeClicked, this);
52829     this.closeBtn.hide();
52830
52831     this.createBody(config);
52832     this.visible = true;
52833     this.collapsed = false;
52834
52835     if(config.hideWhenEmpty){
52836         this.hide();
52837         this.on("paneladded", this.validateVisibility, this);
52838         this.on("panelremoved", this.validateVisibility, this);
52839     }
52840     this.applyConfig(config);
52841 };
52842
52843 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52844
52845     createBody : function(){
52846         /** This region's body element 
52847         * @type Roo.Element */
52848         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52849     },
52850
52851     applyConfig : function(c){
52852         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52853             var dh = Roo.DomHelper;
52854             if(c.titlebar !== false){
52855                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52856                 this.collapseBtn.on("click", this.collapse, this);
52857                 this.collapseBtn.enableDisplayMode();
52858
52859                 if(c.showPin === true || this.showPin){
52860                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52861                     this.stickBtn.enableDisplayMode();
52862                     this.stickBtn.on("click", this.expand, this);
52863                     this.stickBtn.hide();
52864                 }
52865             }
52866             /** This region's collapsed element
52867             * @type Roo.Element */
52868             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52869                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52870             ]}, true);
52871             if(c.floatable !== false){
52872                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52873                this.collapsedEl.on("click", this.collapseClick, this);
52874             }
52875
52876             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52877                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52878                    id: "message", unselectable: "on", style:{"float":"left"}});
52879                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52880              }
52881             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52882             this.expandBtn.on("click", this.expand, this);
52883         }
52884         if(this.collapseBtn){
52885             this.collapseBtn.setVisible(c.collapsible == true);
52886         }
52887         this.cmargins = c.cmargins || this.cmargins ||
52888                          (this.position == "west" || this.position == "east" ?
52889                              {top: 0, left: 2, right:2, bottom: 0} :
52890                              {top: 2, left: 0, right:0, bottom: 2});
52891         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52892         this.bottomTabs = c.tabPosition != "top";
52893         this.autoScroll = c.autoScroll || false;
52894         if(this.autoScroll){
52895             this.bodyEl.setStyle("overflow", "auto");
52896         }else{
52897             this.bodyEl.setStyle("overflow", "hidden");
52898         }
52899         //if(c.titlebar !== false){
52900             if((!c.titlebar && !c.title) || c.titlebar === false){
52901                 this.titleEl.hide();
52902             }else{
52903                 this.titleEl.show();
52904                 if(c.title){
52905                     this.titleTextEl.innerHTML = c.title;
52906                 }
52907             }
52908         //}
52909         this.duration = c.duration || .30;
52910         this.slideDuration = c.slideDuration || .45;
52911         this.config = c;
52912         if(c.collapsed){
52913             this.collapse(true);
52914         }
52915         if(c.hidden){
52916             this.hide();
52917         }
52918     },
52919     /**
52920      * Returns true if this region is currently visible.
52921      * @return {Boolean}
52922      */
52923     isVisible : function(){
52924         return this.visible;
52925     },
52926
52927     /**
52928      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52929      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52930      */
52931     setCollapsedTitle : function(title){
52932         title = title || "&#160;";
52933         if(this.collapsedTitleTextEl){
52934             this.collapsedTitleTextEl.innerHTML = title;
52935         }
52936     },
52937
52938     getBox : function(){
52939         var b;
52940         if(!this.collapsed){
52941             b = this.el.getBox(false, true);
52942         }else{
52943             b = this.collapsedEl.getBox(false, true);
52944         }
52945         return b;
52946     },
52947
52948     getMargins : function(){
52949         return this.collapsed ? this.cmargins : this.margins;
52950     },
52951
52952     highlight : function(){
52953         this.el.addClass("x-layout-panel-dragover");
52954     },
52955
52956     unhighlight : function(){
52957         this.el.removeClass("x-layout-panel-dragover");
52958     },
52959
52960     updateBox : function(box){
52961         this.box = box;
52962         if(!this.collapsed){
52963             this.el.dom.style.left = box.x + "px";
52964             this.el.dom.style.top = box.y + "px";
52965             this.updateBody(box.width, box.height);
52966         }else{
52967             this.collapsedEl.dom.style.left = box.x + "px";
52968             this.collapsedEl.dom.style.top = box.y + "px";
52969             this.collapsedEl.setSize(box.width, box.height);
52970         }
52971         if(this.tabs){
52972             this.tabs.autoSizeTabs();
52973         }
52974     },
52975
52976     updateBody : function(w, h){
52977         if(w !== null){
52978             this.el.setWidth(w);
52979             w -= this.el.getBorderWidth("rl");
52980             if(this.config.adjustments){
52981                 w += this.config.adjustments[0];
52982             }
52983         }
52984         if(h !== null){
52985             this.el.setHeight(h);
52986             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52987             h -= this.el.getBorderWidth("tb");
52988             if(this.config.adjustments){
52989                 h += this.config.adjustments[1];
52990             }
52991             this.bodyEl.setHeight(h);
52992             if(this.tabs){
52993                 h = this.tabs.syncHeight(h);
52994             }
52995         }
52996         if(this.panelSize){
52997             w = w !== null ? w : this.panelSize.width;
52998             h = h !== null ? h : this.panelSize.height;
52999         }
53000         if(this.activePanel){
53001             var el = this.activePanel.getEl();
53002             w = w !== null ? w : el.getWidth();
53003             h = h !== null ? h : el.getHeight();
53004             this.panelSize = {width: w, height: h};
53005             this.activePanel.setSize(w, h);
53006         }
53007         if(Roo.isIE && this.tabs){
53008             this.tabs.el.repaint();
53009         }
53010     },
53011
53012     /**
53013      * Returns the container element for this region.
53014      * @return {Roo.Element}
53015      */
53016     getEl : function(){
53017         return this.el;
53018     },
53019
53020     /**
53021      * Hides this region.
53022      */
53023     hide : function(){
53024         if(!this.collapsed){
53025             this.el.dom.style.left = "-2000px";
53026             this.el.hide();
53027         }else{
53028             this.collapsedEl.dom.style.left = "-2000px";
53029             this.collapsedEl.hide();
53030         }
53031         this.visible = false;
53032         this.fireEvent("visibilitychange", this, false);
53033     },
53034
53035     /**
53036      * Shows this region if it was previously hidden.
53037      */
53038     show : function(){
53039         if(!this.collapsed){
53040             this.el.show();
53041         }else{
53042             this.collapsedEl.show();
53043         }
53044         this.visible = true;
53045         this.fireEvent("visibilitychange", this, true);
53046     },
53047
53048     closeClicked : function(){
53049         if(this.activePanel){
53050             this.remove(this.activePanel);
53051         }
53052     },
53053
53054     collapseClick : function(e){
53055         if(this.isSlid){
53056            e.stopPropagation();
53057            this.slideIn();
53058         }else{
53059            e.stopPropagation();
53060            this.slideOut();
53061         }
53062     },
53063
53064     /**
53065      * Collapses this region.
53066      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53067      */
53068     collapse : function(skipAnim, skipCheck){
53069         if(this.collapsed) {
53070             return;
53071         }
53072         
53073         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53074             
53075             this.collapsed = true;
53076             if(this.split){
53077                 this.split.el.hide();
53078             }
53079             if(this.config.animate && skipAnim !== true){
53080                 this.fireEvent("invalidated", this);
53081                 this.animateCollapse();
53082             }else{
53083                 this.el.setLocation(-20000,-20000);
53084                 this.el.hide();
53085                 this.collapsedEl.show();
53086                 this.fireEvent("collapsed", this);
53087                 this.fireEvent("invalidated", this);
53088             }
53089         }
53090         
53091     },
53092
53093     animateCollapse : function(){
53094         // overridden
53095     },
53096
53097     /**
53098      * Expands this region if it was previously collapsed.
53099      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53100      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53101      */
53102     expand : function(e, skipAnim){
53103         if(e) {
53104             e.stopPropagation();
53105         }
53106         if(!this.collapsed || this.el.hasActiveFx()) {
53107             return;
53108         }
53109         if(this.isSlid){
53110             this.afterSlideIn();
53111             skipAnim = true;
53112         }
53113         this.collapsed = false;
53114         if(this.config.animate && skipAnim !== true){
53115             this.animateExpand();
53116         }else{
53117             this.el.show();
53118             if(this.split){
53119                 this.split.el.show();
53120             }
53121             this.collapsedEl.setLocation(-2000,-2000);
53122             this.collapsedEl.hide();
53123             this.fireEvent("invalidated", this);
53124             this.fireEvent("expanded", this);
53125         }
53126     },
53127
53128     animateExpand : function(){
53129         // overridden
53130     },
53131
53132     initTabs : function()
53133     {
53134         this.bodyEl.setStyle("overflow", "hidden");
53135         var ts = new Roo.TabPanel(
53136                 this.bodyEl.dom,
53137                 {
53138                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53139                     disableTooltips: this.config.disableTabTips,
53140                     toolbar : this.config.toolbar
53141                 }
53142         );
53143         if(this.config.hideTabs){
53144             ts.stripWrap.setDisplayed(false);
53145         }
53146         this.tabs = ts;
53147         ts.resizeTabs = this.config.resizeTabs === true;
53148         ts.minTabWidth = this.config.minTabWidth || 40;
53149         ts.maxTabWidth = this.config.maxTabWidth || 250;
53150         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53151         ts.monitorResize = false;
53152         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53153         ts.bodyEl.addClass('x-layout-tabs-body');
53154         this.panels.each(this.initPanelAsTab, this);
53155     },
53156
53157     initPanelAsTab : function(panel){
53158         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53159                     this.config.closeOnTab && panel.isClosable());
53160         if(panel.tabTip !== undefined){
53161             ti.setTooltip(panel.tabTip);
53162         }
53163         ti.on("activate", function(){
53164               this.setActivePanel(panel);
53165         }, this);
53166         if(this.config.closeOnTab){
53167             ti.on("beforeclose", function(t, e){
53168                 e.cancel = true;
53169                 this.remove(panel);
53170             }, this);
53171         }
53172         return ti;
53173     },
53174
53175     updatePanelTitle : function(panel, title){
53176         if(this.activePanel == panel){
53177             this.updateTitle(title);
53178         }
53179         if(this.tabs){
53180             var ti = this.tabs.getTab(panel.getEl().id);
53181             ti.setText(title);
53182             if(panel.tabTip !== undefined){
53183                 ti.setTooltip(panel.tabTip);
53184             }
53185         }
53186     },
53187
53188     updateTitle : function(title){
53189         if(this.titleTextEl && !this.config.title){
53190             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53191         }
53192     },
53193
53194     setActivePanel : function(panel){
53195         panel = this.getPanel(panel);
53196         if(this.activePanel && this.activePanel != panel){
53197             this.activePanel.setActiveState(false);
53198         }
53199         this.activePanel = panel;
53200         panel.setActiveState(true);
53201         if(this.panelSize){
53202             panel.setSize(this.panelSize.width, this.panelSize.height);
53203         }
53204         if(this.closeBtn){
53205             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53206         }
53207         this.updateTitle(panel.getTitle());
53208         if(this.tabs){
53209             this.fireEvent("invalidated", this);
53210         }
53211         this.fireEvent("panelactivated", this, panel);
53212     },
53213
53214     /**
53215      * Shows the specified panel.
53216      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53217      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53218      */
53219     showPanel : function(panel)
53220     {
53221         panel = this.getPanel(panel);
53222         if(panel){
53223             if(this.tabs){
53224                 var tab = this.tabs.getTab(panel.getEl().id);
53225                 if(tab.isHidden()){
53226                     this.tabs.unhideTab(tab.id);
53227                 }
53228                 tab.activate();
53229             }else{
53230                 this.setActivePanel(panel);
53231             }
53232         }
53233         return panel;
53234     },
53235
53236     /**
53237      * Get the active panel for this region.
53238      * @return {Roo.ContentPanel} The active panel or null
53239      */
53240     getActivePanel : function(){
53241         return this.activePanel;
53242     },
53243
53244     validateVisibility : function(){
53245         if(this.panels.getCount() < 1){
53246             this.updateTitle("&#160;");
53247             this.closeBtn.hide();
53248             this.hide();
53249         }else{
53250             if(!this.isVisible()){
53251                 this.show();
53252             }
53253         }
53254     },
53255
53256     /**
53257      * Adds the passed ContentPanel(s) to this region.
53258      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53259      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53260      */
53261     add : function(panel){
53262         if(arguments.length > 1){
53263             for(var i = 0, len = arguments.length; i < len; i++) {
53264                 this.add(arguments[i]);
53265             }
53266             return null;
53267         }
53268         if(this.hasPanel(panel)){
53269             this.showPanel(panel);
53270             return panel;
53271         }
53272         panel.setRegion(this);
53273         this.panels.add(panel);
53274         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53275             this.bodyEl.dom.appendChild(panel.getEl().dom);
53276             if(panel.background !== true){
53277                 this.setActivePanel(panel);
53278             }
53279             this.fireEvent("paneladded", this, panel);
53280             return panel;
53281         }
53282         if(!this.tabs){
53283             this.initTabs();
53284         }else{
53285             this.initPanelAsTab(panel);
53286         }
53287         if(panel.background !== true){
53288             this.tabs.activate(panel.getEl().id);
53289         }
53290         this.fireEvent("paneladded", this, panel);
53291         return panel;
53292     },
53293
53294     /**
53295      * Hides the tab for the specified panel.
53296      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53297      */
53298     hidePanel : function(panel){
53299         if(this.tabs && (panel = this.getPanel(panel))){
53300             this.tabs.hideTab(panel.getEl().id);
53301         }
53302     },
53303
53304     /**
53305      * Unhides the tab for a previously hidden panel.
53306      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53307      */
53308     unhidePanel : function(panel){
53309         if(this.tabs && (panel = this.getPanel(panel))){
53310             this.tabs.unhideTab(panel.getEl().id);
53311         }
53312     },
53313
53314     clearPanels : function(){
53315         while(this.panels.getCount() > 0){
53316              this.remove(this.panels.first());
53317         }
53318     },
53319
53320     /**
53321      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53322      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53323      * @param {Boolean} preservePanel Overrides the config preservePanel option
53324      * @return {Roo.ContentPanel} The panel that was removed
53325      */
53326     remove : function(panel, preservePanel){
53327         panel = this.getPanel(panel);
53328         if(!panel){
53329             return null;
53330         }
53331         var e = {};
53332         this.fireEvent("beforeremove", this, panel, e);
53333         if(e.cancel === true){
53334             return null;
53335         }
53336         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53337         var panelId = panel.getId();
53338         this.panels.removeKey(panelId);
53339         if(preservePanel){
53340             document.body.appendChild(panel.getEl().dom);
53341         }
53342         if(this.tabs){
53343             this.tabs.removeTab(panel.getEl().id);
53344         }else if (!preservePanel){
53345             this.bodyEl.dom.removeChild(panel.getEl().dom);
53346         }
53347         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53348             var p = this.panels.first();
53349             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53350             tempEl.appendChild(p.getEl().dom);
53351             this.bodyEl.update("");
53352             this.bodyEl.dom.appendChild(p.getEl().dom);
53353             tempEl = null;
53354             this.updateTitle(p.getTitle());
53355             this.tabs = null;
53356             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53357             this.setActivePanel(p);
53358         }
53359         panel.setRegion(null);
53360         if(this.activePanel == panel){
53361             this.activePanel = null;
53362         }
53363         if(this.config.autoDestroy !== false && preservePanel !== true){
53364             try{panel.destroy();}catch(e){}
53365         }
53366         this.fireEvent("panelremoved", this, panel);
53367         return panel;
53368     },
53369
53370     /**
53371      * Returns the TabPanel component used by this region
53372      * @return {Roo.TabPanel}
53373      */
53374     getTabs : function(){
53375         return this.tabs;
53376     },
53377
53378     createTool : function(parentEl, className){
53379         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53380             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53381         btn.addClassOnOver("x-layout-tools-button-over");
53382         return btn;
53383     }
53384 });/*
53385  * Based on:
53386  * Ext JS Library 1.1.1
53387  * Copyright(c) 2006-2007, Ext JS, LLC.
53388  *
53389  * Originally Released Under LGPL - original licence link has changed is not relivant.
53390  *
53391  * Fork - LGPL
53392  * <script type="text/javascript">
53393  */
53394  
53395
53396
53397 /**
53398  * @class Roo.SplitLayoutRegion
53399  * @extends Roo.LayoutRegion
53400  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53401  */
53402 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53403     this.cursor = cursor;
53404     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53405 };
53406
53407 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53408     splitTip : "Drag to resize.",
53409     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53410     useSplitTips : false,
53411
53412     applyConfig : function(config){
53413         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53414         if(config.split){
53415             if(!this.split){
53416                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53417                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53418                 /** The SplitBar for this region 
53419                 * @type Roo.SplitBar */
53420                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53421                 this.split.on("moved", this.onSplitMove, this);
53422                 this.split.useShim = config.useShim === true;
53423                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53424                 if(this.useSplitTips){
53425                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53426                 }
53427                 if(config.collapsible){
53428                     this.split.el.on("dblclick", this.collapse,  this);
53429                 }
53430             }
53431             if(typeof config.minSize != "undefined"){
53432                 this.split.minSize = config.minSize;
53433             }
53434             if(typeof config.maxSize != "undefined"){
53435                 this.split.maxSize = config.maxSize;
53436             }
53437             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53438                 this.hideSplitter();
53439             }
53440         }
53441     },
53442
53443     getHMaxSize : function(){
53444          var cmax = this.config.maxSize || 10000;
53445          var center = this.mgr.getRegion("center");
53446          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53447     },
53448
53449     getVMaxSize : function(){
53450          var cmax = this.config.maxSize || 10000;
53451          var center = this.mgr.getRegion("center");
53452          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53453     },
53454
53455     onSplitMove : function(split, newSize){
53456         this.fireEvent("resized", this, newSize);
53457     },
53458     
53459     /** 
53460      * Returns the {@link Roo.SplitBar} for this region.
53461      * @return {Roo.SplitBar}
53462      */
53463     getSplitBar : function(){
53464         return this.split;
53465     },
53466     
53467     hide : function(){
53468         this.hideSplitter();
53469         Roo.SplitLayoutRegion.superclass.hide.call(this);
53470     },
53471
53472     hideSplitter : function(){
53473         if(this.split){
53474             this.split.el.setLocation(-2000,-2000);
53475             this.split.el.hide();
53476         }
53477     },
53478
53479     show : function(){
53480         if(this.split){
53481             this.split.el.show();
53482         }
53483         Roo.SplitLayoutRegion.superclass.show.call(this);
53484     },
53485     
53486     beforeSlide: function(){
53487         if(Roo.isGecko){// firefox overflow auto bug workaround
53488             this.bodyEl.clip();
53489             if(this.tabs) {
53490                 this.tabs.bodyEl.clip();
53491             }
53492             if(this.activePanel){
53493                 this.activePanel.getEl().clip();
53494                 
53495                 if(this.activePanel.beforeSlide){
53496                     this.activePanel.beforeSlide();
53497                 }
53498             }
53499         }
53500     },
53501     
53502     afterSlide : function(){
53503         if(Roo.isGecko){// firefox overflow auto bug workaround
53504             this.bodyEl.unclip();
53505             if(this.tabs) {
53506                 this.tabs.bodyEl.unclip();
53507             }
53508             if(this.activePanel){
53509                 this.activePanel.getEl().unclip();
53510                 if(this.activePanel.afterSlide){
53511                     this.activePanel.afterSlide();
53512                 }
53513             }
53514         }
53515     },
53516
53517     initAutoHide : function(){
53518         if(this.autoHide !== false){
53519             if(!this.autoHideHd){
53520                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53521                 this.autoHideHd = {
53522                     "mouseout": function(e){
53523                         if(!e.within(this.el, true)){
53524                             st.delay(500);
53525                         }
53526                     },
53527                     "mouseover" : function(e){
53528                         st.cancel();
53529                     },
53530                     scope : this
53531                 };
53532             }
53533             this.el.on(this.autoHideHd);
53534         }
53535     },
53536
53537     clearAutoHide : function(){
53538         if(this.autoHide !== false){
53539             this.el.un("mouseout", this.autoHideHd.mouseout);
53540             this.el.un("mouseover", this.autoHideHd.mouseover);
53541         }
53542     },
53543
53544     clearMonitor : function(){
53545         Roo.get(document).un("click", this.slideInIf, this);
53546     },
53547
53548     // these names are backwards but not changed for compat
53549     slideOut : function(){
53550         if(this.isSlid || this.el.hasActiveFx()){
53551             return;
53552         }
53553         this.isSlid = true;
53554         if(this.collapseBtn){
53555             this.collapseBtn.hide();
53556         }
53557         this.closeBtnState = this.closeBtn.getStyle('display');
53558         this.closeBtn.hide();
53559         if(this.stickBtn){
53560             this.stickBtn.show();
53561         }
53562         this.el.show();
53563         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53564         this.beforeSlide();
53565         this.el.setStyle("z-index", 10001);
53566         this.el.slideIn(this.getSlideAnchor(), {
53567             callback: function(){
53568                 this.afterSlide();
53569                 this.initAutoHide();
53570                 Roo.get(document).on("click", this.slideInIf, this);
53571                 this.fireEvent("slideshow", this);
53572             },
53573             scope: this,
53574             block: true
53575         });
53576     },
53577
53578     afterSlideIn : function(){
53579         this.clearAutoHide();
53580         this.isSlid = false;
53581         this.clearMonitor();
53582         this.el.setStyle("z-index", "");
53583         if(this.collapseBtn){
53584             this.collapseBtn.show();
53585         }
53586         this.closeBtn.setStyle('display', this.closeBtnState);
53587         if(this.stickBtn){
53588             this.stickBtn.hide();
53589         }
53590         this.fireEvent("slidehide", this);
53591     },
53592
53593     slideIn : function(cb){
53594         if(!this.isSlid || this.el.hasActiveFx()){
53595             Roo.callback(cb);
53596             return;
53597         }
53598         this.isSlid = false;
53599         this.beforeSlide();
53600         this.el.slideOut(this.getSlideAnchor(), {
53601             callback: function(){
53602                 this.el.setLeftTop(-10000, -10000);
53603                 this.afterSlide();
53604                 this.afterSlideIn();
53605                 Roo.callback(cb);
53606             },
53607             scope: this,
53608             block: true
53609         });
53610     },
53611     
53612     slideInIf : function(e){
53613         if(!e.within(this.el)){
53614             this.slideIn();
53615         }
53616     },
53617
53618     animateCollapse : function(){
53619         this.beforeSlide();
53620         this.el.setStyle("z-index", 20000);
53621         var anchor = this.getSlideAnchor();
53622         this.el.slideOut(anchor, {
53623             callback : function(){
53624                 this.el.setStyle("z-index", "");
53625                 this.collapsedEl.slideIn(anchor, {duration:.3});
53626                 this.afterSlide();
53627                 this.el.setLocation(-10000,-10000);
53628                 this.el.hide();
53629                 this.fireEvent("collapsed", this);
53630             },
53631             scope: this,
53632             block: true
53633         });
53634     },
53635
53636     animateExpand : function(){
53637         this.beforeSlide();
53638         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53639         this.el.setStyle("z-index", 20000);
53640         this.collapsedEl.hide({
53641             duration:.1
53642         });
53643         this.el.slideIn(this.getSlideAnchor(), {
53644             callback : function(){
53645                 this.el.setStyle("z-index", "");
53646                 this.afterSlide();
53647                 if(this.split){
53648                     this.split.el.show();
53649                 }
53650                 this.fireEvent("invalidated", this);
53651                 this.fireEvent("expanded", this);
53652             },
53653             scope: this,
53654             block: true
53655         });
53656     },
53657
53658     anchors : {
53659         "west" : "left",
53660         "east" : "right",
53661         "north" : "top",
53662         "south" : "bottom"
53663     },
53664
53665     sanchors : {
53666         "west" : "l",
53667         "east" : "r",
53668         "north" : "t",
53669         "south" : "b"
53670     },
53671
53672     canchors : {
53673         "west" : "tl-tr",
53674         "east" : "tr-tl",
53675         "north" : "tl-bl",
53676         "south" : "bl-tl"
53677     },
53678
53679     getAnchor : function(){
53680         return this.anchors[this.position];
53681     },
53682
53683     getCollapseAnchor : function(){
53684         return this.canchors[this.position];
53685     },
53686
53687     getSlideAnchor : function(){
53688         return this.sanchors[this.position];
53689     },
53690
53691     getAlignAdj : function(){
53692         var cm = this.cmargins;
53693         switch(this.position){
53694             case "west":
53695                 return [0, 0];
53696             break;
53697             case "east":
53698                 return [0, 0];
53699             break;
53700             case "north":
53701                 return [0, 0];
53702             break;
53703             case "south":
53704                 return [0, 0];
53705             break;
53706         }
53707     },
53708
53709     getExpandAdj : function(){
53710         var c = this.collapsedEl, cm = this.cmargins;
53711         switch(this.position){
53712             case "west":
53713                 return [-(cm.right+c.getWidth()+cm.left), 0];
53714             break;
53715             case "east":
53716                 return [cm.right+c.getWidth()+cm.left, 0];
53717             break;
53718             case "north":
53719                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53720             break;
53721             case "south":
53722                 return [0, cm.top+cm.bottom+c.getHeight()];
53723             break;
53724         }
53725     }
53726 });/*
53727  * Based on:
53728  * Ext JS Library 1.1.1
53729  * Copyright(c) 2006-2007, Ext JS, LLC.
53730  *
53731  * Originally Released Under LGPL - original licence link has changed is not relivant.
53732  *
53733  * Fork - LGPL
53734  * <script type="text/javascript">
53735  */
53736 /*
53737  * These classes are private internal classes
53738  */
53739 Roo.CenterLayoutRegion = function(mgr, config){
53740     Roo.LayoutRegion.call(this, mgr, config, "center");
53741     this.visible = true;
53742     this.minWidth = config.minWidth || 20;
53743     this.minHeight = config.minHeight || 20;
53744 };
53745
53746 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53747     hide : function(){
53748         // center panel can't be hidden
53749     },
53750     
53751     show : function(){
53752         // center panel can't be hidden
53753     },
53754     
53755     getMinWidth: function(){
53756         return this.minWidth;
53757     },
53758     
53759     getMinHeight: function(){
53760         return this.minHeight;
53761     }
53762 });
53763
53764
53765 Roo.NorthLayoutRegion = function(mgr, config){
53766     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53767     if(this.split){
53768         this.split.placement = Roo.SplitBar.TOP;
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.NorthLayoutRegion, 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             box.height += this.split.el.getHeight();
53786         }
53787         return box;
53788     },
53789     
53790     updateBox : function(box){
53791         if(this.split && !this.collapsed){
53792             box.height -= this.split.el.getHeight();
53793             this.split.el.setLeft(box.x);
53794             this.split.el.setTop(box.y+box.height);
53795             this.split.el.setWidth(box.width);
53796         }
53797         if(this.collapsed){
53798             this.updateBody(box.width, null);
53799         }
53800         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53801     }
53802 });
53803
53804 Roo.SouthLayoutRegion = function(mgr, config){
53805     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53806     if(this.split){
53807         this.split.placement = Roo.SplitBar.BOTTOM;
53808         this.split.orientation = Roo.SplitBar.VERTICAL;
53809         this.split.el.addClass("x-layout-split-v");
53810     }
53811     var size = config.initialSize || config.height;
53812     if(typeof size != "undefined"){
53813         this.el.setHeight(size);
53814     }
53815 };
53816 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53817     orientation: Roo.SplitBar.VERTICAL,
53818     getBox : function(){
53819         if(this.collapsed){
53820             return this.collapsedEl.getBox();
53821         }
53822         var box = this.el.getBox();
53823         if(this.split){
53824             var sh = this.split.el.getHeight();
53825             box.height += sh;
53826             box.y -= sh;
53827         }
53828         return box;
53829     },
53830     
53831     updateBox : function(box){
53832         if(this.split && !this.collapsed){
53833             var sh = this.split.el.getHeight();
53834             box.height -= sh;
53835             box.y += sh;
53836             this.split.el.setLeft(box.x);
53837             this.split.el.setTop(box.y-sh);
53838             this.split.el.setWidth(box.width);
53839         }
53840         if(this.collapsed){
53841             this.updateBody(box.width, null);
53842         }
53843         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53844     }
53845 });
53846
53847 Roo.EastLayoutRegion = function(mgr, config){
53848     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53849     if(this.split){
53850         this.split.placement = Roo.SplitBar.RIGHT;
53851         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53852         this.split.el.addClass("x-layout-split-h");
53853     }
53854     var size = config.initialSize || config.width;
53855     if(typeof size != "undefined"){
53856         this.el.setWidth(size);
53857     }
53858 };
53859 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53860     orientation: Roo.SplitBar.HORIZONTAL,
53861     getBox : function(){
53862         if(this.collapsed){
53863             return this.collapsedEl.getBox();
53864         }
53865         var box = this.el.getBox();
53866         if(this.split){
53867             var sw = this.split.el.getWidth();
53868             box.width += sw;
53869             box.x -= sw;
53870         }
53871         return box;
53872     },
53873
53874     updateBox : function(box){
53875         if(this.split && !this.collapsed){
53876             var sw = this.split.el.getWidth();
53877             box.width -= sw;
53878             this.split.el.setLeft(box.x);
53879             this.split.el.setTop(box.y);
53880             this.split.el.setHeight(box.height);
53881             box.x += sw;
53882         }
53883         if(this.collapsed){
53884             this.updateBody(null, box.height);
53885         }
53886         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53887     }
53888 });
53889
53890 Roo.WestLayoutRegion = function(mgr, config){
53891     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53892     if(this.split){
53893         this.split.placement = Roo.SplitBar.LEFT;
53894         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53895         this.split.el.addClass("x-layout-split-h");
53896     }
53897     var size = config.initialSize || config.width;
53898     if(typeof size != "undefined"){
53899         this.el.setWidth(size);
53900     }
53901 };
53902 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53903     orientation: Roo.SplitBar.HORIZONTAL,
53904     getBox : function(){
53905         if(this.collapsed){
53906             return this.collapsedEl.getBox();
53907         }
53908         var box = this.el.getBox();
53909         if(this.split){
53910             box.width += this.split.el.getWidth();
53911         }
53912         return box;
53913     },
53914     
53915     updateBox : function(box){
53916         if(this.split && !this.collapsed){
53917             var sw = this.split.el.getWidth();
53918             box.width -= sw;
53919             this.split.el.setLeft(box.x+box.width);
53920             this.split.el.setTop(box.y);
53921             this.split.el.setHeight(box.height);
53922         }
53923         if(this.collapsed){
53924             this.updateBody(null, box.height);
53925         }
53926         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53927     }
53928 });
53929 /*
53930  * Based on:
53931  * Ext JS Library 1.1.1
53932  * Copyright(c) 2006-2007, Ext JS, LLC.
53933  *
53934  * Originally Released Under LGPL - original licence link has changed is not relivant.
53935  *
53936  * Fork - LGPL
53937  * <script type="text/javascript">
53938  */
53939  
53940  
53941 /*
53942  * Private internal class for reading and applying state
53943  */
53944 Roo.LayoutStateManager = function(layout){
53945      // default empty state
53946      this.state = {
53947         north: {},
53948         south: {},
53949         east: {},
53950         west: {}       
53951     };
53952 };
53953
53954 Roo.LayoutStateManager.prototype = {
53955     init : function(layout, provider){
53956         this.provider = provider;
53957         var state = provider.get(layout.id+"-layout-state");
53958         if(state){
53959             var wasUpdating = layout.isUpdating();
53960             if(!wasUpdating){
53961                 layout.beginUpdate();
53962             }
53963             for(var key in state){
53964                 if(typeof state[key] != "function"){
53965                     var rstate = state[key];
53966                     var r = layout.getRegion(key);
53967                     if(r && rstate){
53968                         if(rstate.size){
53969                             r.resizeTo(rstate.size);
53970                         }
53971                         if(rstate.collapsed == true){
53972                             r.collapse(true);
53973                         }else{
53974                             r.expand(null, true);
53975                         }
53976                     }
53977                 }
53978             }
53979             if(!wasUpdating){
53980                 layout.endUpdate();
53981             }
53982             this.state = state; 
53983         }
53984         this.layout = layout;
53985         layout.on("regionresized", this.onRegionResized, this);
53986         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53987         layout.on("regionexpanded", this.onRegionExpanded, this);
53988     },
53989     
53990     storeState : function(){
53991         this.provider.set(this.layout.id+"-layout-state", this.state);
53992     },
53993     
53994     onRegionResized : function(region, newSize){
53995         this.state[region.getPosition()].size = newSize;
53996         this.storeState();
53997     },
53998     
53999     onRegionCollapsed : function(region){
54000         this.state[region.getPosition()].collapsed = true;
54001         this.storeState();
54002     },
54003     
54004     onRegionExpanded : function(region){
54005         this.state[region.getPosition()].collapsed = false;
54006         this.storeState();
54007     }
54008 };/*
54009  * Based on:
54010  * Ext JS Library 1.1.1
54011  * Copyright(c) 2006-2007, Ext JS, LLC.
54012  *
54013  * Originally Released Under LGPL - original licence link has changed is not relivant.
54014  *
54015  * Fork - LGPL
54016  * <script type="text/javascript">
54017  */
54018 /**
54019  * @class Roo.ContentPanel
54020  * @extends Roo.util.Observable
54021  * A basic ContentPanel element.
54022  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
54023  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
54024  * @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
54025  * @cfg {Boolean}   closable      True if the panel can be closed/removed
54026  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
54027  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
54028  * @cfg {Toolbar}   toolbar       A toolbar for this panel
54029  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
54030  * @cfg {String} title          The title for this panel
54031  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
54032  * @cfg {String} url            Calls {@link #setUrl} with this value
54033  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
54034  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
54035  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
54036  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
54037
54038  * @constructor
54039  * Create a new ContentPanel.
54040  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
54041  * @param {String/Object} config A string to set only the title or a config object
54042  * @param {String} content (optional) Set the HTML content for this panel
54043  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
54044  */
54045 Roo.ContentPanel = function(el, config, content){
54046     
54047      
54048     /*
54049     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
54050         config = el;
54051         el = Roo.id();
54052     }
54053     if (config && config.parentLayout) { 
54054         el = config.parentLayout.el.createChild(); 
54055     }
54056     */
54057     if(el.autoCreate){ // xtype is available if this is called from factory
54058         config = el;
54059         el = Roo.id();
54060     }
54061     this.el = Roo.get(el);
54062     if(!this.el && config && config.autoCreate){
54063         if(typeof config.autoCreate == "object"){
54064             if(!config.autoCreate.id){
54065                 config.autoCreate.id = config.id||el;
54066             }
54067             this.el = Roo.DomHelper.append(document.body,
54068                         config.autoCreate, true);
54069         }else{
54070             this.el = Roo.DomHelper.append(document.body,
54071                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54072         }
54073     }
54074     this.closable = false;
54075     this.loaded = false;
54076     this.active = false;
54077     if(typeof config == "string"){
54078         this.title = config;
54079     }else{
54080         Roo.apply(this, config);
54081     }
54082     
54083     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54084         this.wrapEl = this.el.wrap();
54085         this.toolbar.container = this.el.insertSibling(false, 'before');
54086         this.toolbar = new Roo.Toolbar(this.toolbar);
54087     }
54088     
54089     // xtype created footer. - not sure if will work as we normally have to render first..
54090     if (this.footer && !this.footer.el && this.footer.xtype) {
54091         if (!this.wrapEl) {
54092             this.wrapEl = this.el.wrap();
54093         }
54094     
54095         this.footer.container = this.wrapEl.createChild();
54096          
54097         this.footer = Roo.factory(this.footer, Roo);
54098         
54099     }
54100     
54101     if(this.resizeEl){
54102         this.resizeEl = Roo.get(this.resizeEl, true);
54103     }else{
54104         this.resizeEl = this.el;
54105     }
54106     // handle view.xtype
54107     
54108  
54109     
54110     
54111     this.addEvents({
54112         /**
54113          * @event activate
54114          * Fires when this panel is activated. 
54115          * @param {Roo.ContentPanel} this
54116          */
54117         "activate" : true,
54118         /**
54119          * @event deactivate
54120          * Fires when this panel is activated. 
54121          * @param {Roo.ContentPanel} this
54122          */
54123         "deactivate" : true,
54124
54125         /**
54126          * @event resize
54127          * Fires when this panel is resized if fitToFrame is true.
54128          * @param {Roo.ContentPanel} this
54129          * @param {Number} width The width after any component adjustments
54130          * @param {Number} height The height after any component adjustments
54131          */
54132         "resize" : true,
54133         
54134          /**
54135          * @event render
54136          * Fires when this tab is created
54137          * @param {Roo.ContentPanel} this
54138          */
54139         "render" : true
54140          
54141         
54142     });
54143     
54144
54145     
54146     
54147     if(this.autoScroll){
54148         this.resizeEl.setStyle("overflow", "auto");
54149     } else {
54150         // fix randome scrolling
54151         this.el.on('scroll', function() {
54152             Roo.log('fix random scolling');
54153             this.scrollTo('top',0); 
54154         });
54155     }
54156     content = content || this.content;
54157     if(content){
54158         this.setContent(content);
54159     }
54160     if(config && config.url){
54161         this.setUrl(this.url, this.params, this.loadOnce);
54162     }
54163     
54164     
54165     
54166     Roo.ContentPanel.superclass.constructor.call(this);
54167     
54168     if (this.view && typeof(this.view.xtype) != 'undefined') {
54169         this.view.el = this.el.appendChild(document.createElement("div"));
54170         this.view = Roo.factory(this.view); 
54171         this.view.render  &&  this.view.render(false, '');  
54172     }
54173     
54174     
54175     this.fireEvent('render', this);
54176 };
54177
54178 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54179     tabTip:'',
54180     setRegion : function(region){
54181         this.region = region;
54182         if(region){
54183            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54184         }else{
54185            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54186         } 
54187     },
54188     
54189     /**
54190      * Returns the toolbar for this Panel if one was configured. 
54191      * @return {Roo.Toolbar} 
54192      */
54193     getToolbar : function(){
54194         return this.toolbar;
54195     },
54196     
54197     setActiveState : function(active){
54198         this.active = active;
54199         if(!active){
54200             this.fireEvent("deactivate", this);
54201         }else{
54202             this.fireEvent("activate", this);
54203         }
54204     },
54205     /**
54206      * Updates this panel's element
54207      * @param {String} content The new content
54208      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54209     */
54210     setContent : function(content, loadScripts){
54211         this.el.update(content, loadScripts);
54212     },
54213
54214     ignoreResize : function(w, h){
54215         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54216             return true;
54217         }else{
54218             this.lastSize = {width: w, height: h};
54219             return false;
54220         }
54221     },
54222     /**
54223      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54224      * @return {Roo.UpdateManager} The UpdateManager
54225      */
54226     getUpdateManager : function(){
54227         return this.el.getUpdateManager();
54228     },
54229      /**
54230      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54231      * @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:
54232 <pre><code>
54233 panel.load({
54234     url: "your-url.php",
54235     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54236     callback: yourFunction,
54237     scope: yourObject, //(optional scope)
54238     discardUrl: false,
54239     nocache: false,
54240     text: "Loading...",
54241     timeout: 30,
54242     scripts: false
54243 });
54244 </code></pre>
54245      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54246      * 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.
54247      * @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}
54248      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54249      * @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.
54250      * @return {Roo.ContentPanel} this
54251      */
54252     load : function(){
54253         var um = this.el.getUpdateManager();
54254         um.update.apply(um, arguments);
54255         return this;
54256     },
54257
54258
54259     /**
54260      * 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.
54261      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54262      * @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)
54263      * @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)
54264      * @return {Roo.UpdateManager} The UpdateManager
54265      */
54266     setUrl : function(url, params, loadOnce){
54267         if(this.refreshDelegate){
54268             this.removeListener("activate", this.refreshDelegate);
54269         }
54270         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54271         this.on("activate", this.refreshDelegate);
54272         return this.el.getUpdateManager();
54273     },
54274     
54275     _handleRefresh : function(url, params, loadOnce){
54276         if(!loadOnce || !this.loaded){
54277             var updater = this.el.getUpdateManager();
54278             updater.update(url, params, this._setLoaded.createDelegate(this));
54279         }
54280     },
54281     
54282     _setLoaded : function(){
54283         this.loaded = true;
54284     }, 
54285     
54286     /**
54287      * Returns this panel's id
54288      * @return {String} 
54289      */
54290     getId : function(){
54291         return this.el.id;
54292     },
54293     
54294     /** 
54295      * Returns this panel's element - used by regiosn to add.
54296      * @return {Roo.Element} 
54297      */
54298     getEl : function(){
54299         return this.wrapEl || this.el;
54300     },
54301     
54302     adjustForComponents : function(width, height)
54303     {
54304         //Roo.log('adjustForComponents ');
54305         if(this.resizeEl != this.el){
54306             width -= this.el.getFrameWidth('lr');
54307             height -= this.el.getFrameWidth('tb');
54308         }
54309         if(this.toolbar){
54310             var te = this.toolbar.getEl();
54311             height -= te.getHeight();
54312             te.setWidth(width);
54313         }
54314         if(this.footer){
54315             var te = this.footer.getEl();
54316             //Roo.log("footer:" + te.getHeight());
54317             
54318             height -= te.getHeight();
54319             te.setWidth(width);
54320         }
54321         
54322         
54323         if(this.adjustments){
54324             width += this.adjustments[0];
54325             height += this.adjustments[1];
54326         }
54327         return {"width": width, "height": height};
54328     },
54329     
54330     setSize : function(width, height){
54331         if(this.fitToFrame && !this.ignoreResize(width, height)){
54332             if(this.fitContainer && this.resizeEl != this.el){
54333                 this.el.setSize(width, height);
54334             }
54335             var size = this.adjustForComponents(width, height);
54336             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54337             this.fireEvent('resize', this, size.width, size.height);
54338         }
54339     },
54340     
54341     /**
54342      * Returns this panel's title
54343      * @return {String} 
54344      */
54345     getTitle : function(){
54346         return this.title;
54347     },
54348     
54349     /**
54350      * Set this panel's title
54351      * @param {String} title
54352      */
54353     setTitle : function(title){
54354         this.title = title;
54355         if(this.region){
54356             this.region.updatePanelTitle(this, title);
54357         }
54358     },
54359     
54360     /**
54361      * Returns true is this panel was configured to be closable
54362      * @return {Boolean} 
54363      */
54364     isClosable : function(){
54365         return this.closable;
54366     },
54367     
54368     beforeSlide : function(){
54369         this.el.clip();
54370         this.resizeEl.clip();
54371     },
54372     
54373     afterSlide : function(){
54374         this.el.unclip();
54375         this.resizeEl.unclip();
54376     },
54377     
54378     /**
54379      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54380      *   Will fail silently if the {@link #setUrl} method has not been called.
54381      *   This does not activate the panel, just updates its content.
54382      */
54383     refresh : function(){
54384         if(this.refreshDelegate){
54385            this.loaded = false;
54386            this.refreshDelegate();
54387         }
54388     },
54389     
54390     /**
54391      * Destroys this panel
54392      */
54393     destroy : function(){
54394         this.el.removeAllListeners();
54395         var tempEl = document.createElement("span");
54396         tempEl.appendChild(this.el.dom);
54397         tempEl.innerHTML = "";
54398         this.el.remove();
54399         this.el = null;
54400     },
54401     
54402     /**
54403      * form - if the content panel contains a form - this is a reference to it.
54404      * @type {Roo.form.Form}
54405      */
54406     form : false,
54407     /**
54408      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54409      *    This contains a reference to it.
54410      * @type {Roo.View}
54411      */
54412     view : false,
54413     
54414       /**
54415      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54416      * <pre><code>
54417
54418 layout.addxtype({
54419        xtype : 'Form',
54420        items: [ .... ]
54421    }
54422 );
54423
54424 </code></pre>
54425      * @param {Object} cfg Xtype definition of item to add.
54426      */
54427     
54428     addxtype : function(cfg) {
54429         // add form..
54430         if (cfg.xtype.match(/^Form$/)) {
54431             
54432             var el;
54433             //if (this.footer) {
54434             //    el = this.footer.container.insertSibling(false, 'before');
54435             //} else {
54436                 el = this.el.createChild();
54437             //}
54438
54439             this.form = new  Roo.form.Form(cfg);
54440             
54441             
54442             if ( this.form.allItems.length) {
54443                 this.form.render(el.dom);
54444             }
54445             return this.form;
54446         }
54447         // should only have one of theses..
54448         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54449             // views.. should not be just added - used named prop 'view''
54450             
54451             cfg.el = this.el.appendChild(document.createElement("div"));
54452             // factory?
54453             
54454             var ret = new Roo.factory(cfg);
54455              
54456              ret.render && ret.render(false, ''); // render blank..
54457             this.view = ret;
54458             return ret;
54459         }
54460         return false;
54461     }
54462 });
54463
54464 /**
54465  * @class Roo.GridPanel
54466  * @extends Roo.ContentPanel
54467  * @constructor
54468  * Create a new GridPanel.
54469  * @param {Roo.grid.Grid} grid The grid for this panel
54470  * @param {String/Object} config A string to set only the panel's title, or a config object
54471  */
54472 Roo.GridPanel = function(grid, config){
54473     
54474   
54475     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54476         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54477         
54478     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54479     
54480     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54481     
54482     if(this.toolbar){
54483         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54484     }
54485     // xtype created footer. - not sure if will work as we normally have to render first..
54486     if (this.footer && !this.footer.el && this.footer.xtype) {
54487         
54488         this.footer.container = this.grid.getView().getFooterPanel(true);
54489         this.footer.dataSource = this.grid.dataSource;
54490         this.footer = Roo.factory(this.footer, Roo);
54491         
54492     }
54493     
54494     grid.monitorWindowResize = false; // turn off autosizing
54495     grid.autoHeight = false;
54496     grid.autoWidth = false;
54497     this.grid = grid;
54498     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54499 };
54500
54501 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54502     getId : function(){
54503         return this.grid.id;
54504     },
54505     
54506     /**
54507      * Returns the grid for this panel
54508      * @return {Roo.grid.Grid} 
54509      */
54510     getGrid : function(){
54511         return this.grid;    
54512     },
54513     
54514     setSize : function(width, height){
54515         if(!this.ignoreResize(width, height)){
54516             var grid = this.grid;
54517             var size = this.adjustForComponents(width, height);
54518             grid.getGridEl().setSize(size.width, size.height);
54519             grid.autoSize();
54520         }
54521     },
54522     
54523     beforeSlide : function(){
54524         this.grid.getView().scroller.clip();
54525     },
54526     
54527     afterSlide : function(){
54528         this.grid.getView().scroller.unclip();
54529     },
54530     
54531     destroy : function(){
54532         this.grid.destroy();
54533         delete this.grid;
54534         Roo.GridPanel.superclass.destroy.call(this); 
54535     }
54536 });
54537
54538
54539 /**
54540  * @class Roo.NestedLayoutPanel
54541  * @extends Roo.ContentPanel
54542  * @constructor
54543  * Create a new NestedLayoutPanel.
54544  * 
54545  * 
54546  * @param {Roo.BorderLayout} layout The layout for this panel
54547  * @param {String/Object} config A string to set only the title or a config object
54548  */
54549 Roo.NestedLayoutPanel = function(layout, config)
54550 {
54551     // construct with only one argument..
54552     /* FIXME - implement nicer consturctors
54553     if (layout.layout) {
54554         config = layout;
54555         layout = config.layout;
54556         delete config.layout;
54557     }
54558     if (layout.xtype && !layout.getEl) {
54559         // then layout needs constructing..
54560         layout = Roo.factory(layout, Roo);
54561     }
54562     */
54563     
54564     
54565     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54566     
54567     layout.monitorWindowResize = false; // turn off autosizing
54568     this.layout = layout;
54569     this.layout.getEl().addClass("x-layout-nested-layout");
54570     
54571     
54572     
54573     
54574 };
54575
54576 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54577
54578     setSize : function(width, height){
54579         if(!this.ignoreResize(width, height)){
54580             var size = this.adjustForComponents(width, height);
54581             var el = this.layout.getEl();
54582             el.setSize(size.width, size.height);
54583             var touch = el.dom.offsetWidth;
54584             this.layout.layout();
54585             // ie requires a double layout on the first pass
54586             if(Roo.isIE && !this.initialized){
54587                 this.initialized = true;
54588                 this.layout.layout();
54589             }
54590         }
54591     },
54592     
54593     // activate all subpanels if not currently active..
54594     
54595     setActiveState : function(active){
54596         this.active = active;
54597         if(!active){
54598             this.fireEvent("deactivate", this);
54599             return;
54600         }
54601         
54602         this.fireEvent("activate", this);
54603         // not sure if this should happen before or after..
54604         if (!this.layout) {
54605             return; // should not happen..
54606         }
54607         var reg = false;
54608         for (var r in this.layout.regions) {
54609             reg = this.layout.getRegion(r);
54610             if (reg.getActivePanel()) {
54611                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54612                 reg.setActivePanel(reg.getActivePanel());
54613                 continue;
54614             }
54615             if (!reg.panels.length) {
54616                 continue;
54617             }
54618             reg.showPanel(reg.getPanel(0));
54619         }
54620         
54621         
54622         
54623         
54624     },
54625     
54626     /**
54627      * Returns the nested BorderLayout for this panel
54628      * @return {Roo.BorderLayout} 
54629      */
54630     getLayout : function(){
54631         return this.layout;
54632     },
54633     
54634      /**
54635      * Adds a xtype elements to the layout of the nested panel
54636      * <pre><code>
54637
54638 panel.addxtype({
54639        xtype : 'ContentPanel',
54640        region: 'west',
54641        items: [ .... ]
54642    }
54643 );
54644
54645 panel.addxtype({
54646         xtype : 'NestedLayoutPanel',
54647         region: 'west',
54648         layout: {
54649            center: { },
54650            west: { }   
54651         },
54652         items : [ ... list of content panels or nested layout panels.. ]
54653    }
54654 );
54655 </code></pre>
54656      * @param {Object} cfg Xtype definition of item to add.
54657      */
54658     addxtype : function(cfg) {
54659         return this.layout.addxtype(cfg);
54660     
54661     }
54662 });
54663
54664 Roo.ScrollPanel = function(el, config, content){
54665     config = config || {};
54666     config.fitToFrame = true;
54667     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54668     
54669     this.el.dom.style.overflow = "hidden";
54670     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54671     this.el.removeClass("x-layout-inactive-content");
54672     this.el.on("mousewheel", this.onWheel, this);
54673
54674     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54675     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54676     up.unselectable(); down.unselectable();
54677     up.on("click", this.scrollUp, this);
54678     down.on("click", this.scrollDown, this);
54679     up.addClassOnOver("x-scroller-btn-over");
54680     down.addClassOnOver("x-scroller-btn-over");
54681     up.addClassOnClick("x-scroller-btn-click");
54682     down.addClassOnClick("x-scroller-btn-click");
54683     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54684
54685     this.resizeEl = this.el;
54686     this.el = wrap; this.up = up; this.down = down;
54687 };
54688
54689 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54690     increment : 100,
54691     wheelIncrement : 5,
54692     scrollUp : function(){
54693         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54694     },
54695
54696     scrollDown : function(){
54697         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54698     },
54699
54700     afterScroll : function(){
54701         var el = this.resizeEl;
54702         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54703         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54704         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54705     },
54706
54707     setSize : function(){
54708         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54709         this.afterScroll();
54710     },
54711
54712     onWheel : function(e){
54713         var d = e.getWheelDelta();
54714         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54715         this.afterScroll();
54716         e.stopEvent();
54717     },
54718
54719     setContent : function(content, loadScripts){
54720         this.resizeEl.update(content, loadScripts);
54721     }
54722
54723 });
54724
54725
54726
54727
54728
54729
54730
54731
54732
54733 /**
54734  * @class Roo.TreePanel
54735  * @extends Roo.ContentPanel
54736  * @constructor
54737  * Create a new TreePanel. - defaults to fit/scoll contents.
54738  * @param {String/Object} config A string to set only the panel's title, or a config object
54739  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54740  */
54741 Roo.TreePanel = function(config){
54742     var el = config.el;
54743     var tree = config.tree;
54744     delete config.tree; 
54745     delete config.el; // hopefull!
54746     
54747     // wrapper for IE7 strict & safari scroll issue
54748     
54749     var treeEl = el.createChild();
54750     config.resizeEl = treeEl;
54751     
54752     
54753     
54754     Roo.TreePanel.superclass.constructor.call(this, el, config);
54755  
54756  
54757     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54758     //console.log(tree);
54759     this.on('activate', function()
54760     {
54761         if (this.tree.rendered) {
54762             return;
54763         }
54764         //console.log('render tree');
54765         this.tree.render();
54766     });
54767     // this should not be needed.. - it's actually the 'el' that resizes?
54768     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54769     
54770     //this.on('resize',  function (cp, w, h) {
54771     //        this.tree.innerCt.setWidth(w);
54772     //        this.tree.innerCt.setHeight(h);
54773     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54774     //});
54775
54776         
54777     
54778 };
54779
54780 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54781     fitToFrame : true,
54782     autoScroll : true
54783 });
54784
54785
54786
54787
54788
54789
54790
54791
54792
54793
54794
54795 /*
54796  * Based on:
54797  * Ext JS Library 1.1.1
54798  * Copyright(c) 2006-2007, Ext JS, LLC.
54799  *
54800  * Originally Released Under LGPL - original licence link has changed is not relivant.
54801  *
54802  * Fork - LGPL
54803  * <script type="text/javascript">
54804  */
54805  
54806
54807 /**
54808  * @class Roo.ReaderLayout
54809  * @extends Roo.BorderLayout
54810  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54811  * center region containing two nested regions (a top one for a list view and one for item preview below),
54812  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54813  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54814  * expedites the setup of the overall layout and regions for this common application style.
54815  * Example:
54816  <pre><code>
54817 var reader = new Roo.ReaderLayout();
54818 var CP = Roo.ContentPanel;  // shortcut for adding
54819
54820 reader.beginUpdate();
54821 reader.add("north", new CP("north", "North"));
54822 reader.add("west", new CP("west", {title: "West"}));
54823 reader.add("east", new CP("east", {title: "East"}));
54824
54825 reader.regions.listView.add(new CP("listView", "List"));
54826 reader.regions.preview.add(new CP("preview", "Preview"));
54827 reader.endUpdate();
54828 </code></pre>
54829 * @constructor
54830 * Create a new ReaderLayout
54831 * @param {Object} config Configuration options
54832 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54833 * document.body if omitted)
54834 */
54835 Roo.ReaderLayout = function(config, renderTo){
54836     var c = config || {size:{}};
54837     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54838         north: c.north !== false ? Roo.apply({
54839             split:false,
54840             initialSize: 32,
54841             titlebar: false
54842         }, c.north) : false,
54843         west: c.west !== false ? Roo.apply({
54844             split:true,
54845             initialSize: 200,
54846             minSize: 175,
54847             maxSize: 400,
54848             titlebar: true,
54849             collapsible: true,
54850             animate: true,
54851             margins:{left:5,right:0,bottom:5,top:5},
54852             cmargins:{left:5,right:5,bottom:5,top:5}
54853         }, c.west) : false,
54854         east: c.east !== false ? Roo.apply({
54855             split:true,
54856             initialSize: 200,
54857             minSize: 175,
54858             maxSize: 400,
54859             titlebar: true,
54860             collapsible: true,
54861             animate: true,
54862             margins:{left:0,right:5,bottom:5,top:5},
54863             cmargins:{left:5,right:5,bottom:5,top:5}
54864         }, c.east) : false,
54865         center: Roo.apply({
54866             tabPosition: 'top',
54867             autoScroll:false,
54868             closeOnTab: true,
54869             titlebar:false,
54870             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54871         }, c.center)
54872     });
54873
54874     this.el.addClass('x-reader');
54875
54876     this.beginUpdate();
54877
54878     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54879         south: c.preview !== false ? Roo.apply({
54880             split:true,
54881             initialSize: 200,
54882             minSize: 100,
54883             autoScroll:true,
54884             collapsible:true,
54885             titlebar: true,
54886             cmargins:{top:5,left:0, right:0, bottom:0}
54887         }, c.preview) : false,
54888         center: Roo.apply({
54889             autoScroll:false,
54890             titlebar:false,
54891             minHeight:200
54892         }, c.listView)
54893     });
54894     this.add('center', new Roo.NestedLayoutPanel(inner,
54895             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54896
54897     this.endUpdate();
54898
54899     this.regions.preview = inner.getRegion('south');
54900     this.regions.listView = inner.getRegion('center');
54901 };
54902
54903 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54904  * Based on:
54905  * Ext JS Library 1.1.1
54906  * Copyright(c) 2006-2007, Ext JS, LLC.
54907  *
54908  * Originally Released Under LGPL - original licence link has changed is not relivant.
54909  *
54910  * Fork - LGPL
54911  * <script type="text/javascript">
54912  */
54913  
54914 /**
54915  * @class Roo.grid.Grid
54916  * @extends Roo.util.Observable
54917  * This class represents the primary interface of a component based grid control.
54918  * <br><br>Usage:<pre><code>
54919  var grid = new Roo.grid.Grid("my-container-id", {
54920      ds: myDataStore,
54921      cm: myColModel,
54922      selModel: mySelectionModel,
54923      autoSizeColumns: true,
54924      monitorWindowResize: false,
54925      trackMouseOver: true
54926  });
54927  // set any options
54928  grid.render();
54929  * </code></pre>
54930  * <b>Common Problems:</b><br/>
54931  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54932  * element will correct this<br/>
54933  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54934  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54935  * are unpredictable.<br/>
54936  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54937  * grid to calculate dimensions/offsets.<br/>
54938   * @constructor
54939  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54940  * The container MUST have some type of size defined for the grid to fill. The container will be
54941  * automatically set to position relative if it isn't already.
54942  * @param {Object} config A config object that sets properties on this grid.
54943  */
54944 Roo.grid.Grid = function(container, config){
54945         // initialize the container
54946         this.container = Roo.get(container);
54947         this.container.update("");
54948         this.container.setStyle("overflow", "hidden");
54949     this.container.addClass('x-grid-container');
54950
54951     this.id = this.container.id;
54952
54953     Roo.apply(this, config);
54954     // check and correct shorthanded configs
54955     if(this.ds){
54956         this.dataSource = this.ds;
54957         delete this.ds;
54958     }
54959     if(this.cm){
54960         this.colModel = this.cm;
54961         delete this.cm;
54962     }
54963     if(this.sm){
54964         this.selModel = this.sm;
54965         delete this.sm;
54966     }
54967
54968     if (this.selModel) {
54969         this.selModel = Roo.factory(this.selModel, Roo.grid);
54970         this.sm = this.selModel;
54971         this.sm.xmodule = this.xmodule || false;
54972     }
54973     if (typeof(this.colModel.config) == 'undefined') {
54974         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54975         this.cm = this.colModel;
54976         this.cm.xmodule = this.xmodule || false;
54977     }
54978     if (this.dataSource) {
54979         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54980         this.ds = this.dataSource;
54981         this.ds.xmodule = this.xmodule || false;
54982          
54983     }
54984     
54985     
54986     
54987     if(this.width){
54988         this.container.setWidth(this.width);
54989     }
54990
54991     if(this.height){
54992         this.container.setHeight(this.height);
54993     }
54994     /** @private */
54995         this.addEvents({
54996         // raw events
54997         /**
54998          * @event click
54999          * The raw click event for the entire grid.
55000          * @param {Roo.EventObject} e
55001          */
55002         "click" : true,
55003         /**
55004          * @event dblclick
55005          * The raw dblclick event for the entire grid.
55006          * @param {Roo.EventObject} e
55007          */
55008         "dblclick" : true,
55009         /**
55010          * @event contextmenu
55011          * The raw contextmenu event for the entire grid.
55012          * @param {Roo.EventObject} e
55013          */
55014         "contextmenu" : true,
55015         /**
55016          * @event mousedown
55017          * The raw mousedown event for the entire grid.
55018          * @param {Roo.EventObject} e
55019          */
55020         "mousedown" : true,
55021         /**
55022          * @event mouseup
55023          * The raw mouseup event for the entire grid.
55024          * @param {Roo.EventObject} e
55025          */
55026         "mouseup" : true,
55027         /**
55028          * @event mouseover
55029          * The raw mouseover event for the entire grid.
55030          * @param {Roo.EventObject} e
55031          */
55032         "mouseover" : true,
55033         /**
55034          * @event mouseout
55035          * The raw mouseout event for the entire grid.
55036          * @param {Roo.EventObject} e
55037          */
55038         "mouseout" : true,
55039         /**
55040          * @event keypress
55041          * The raw keypress event for the entire grid.
55042          * @param {Roo.EventObject} e
55043          */
55044         "keypress" : true,
55045         /**
55046          * @event keydown
55047          * The raw keydown event for the entire grid.
55048          * @param {Roo.EventObject} e
55049          */
55050         "keydown" : true,
55051
55052         // custom events
55053
55054         /**
55055          * @event cellclick
55056          * Fires when a cell is clicked
55057          * @param {Grid} this
55058          * @param {Number} rowIndex
55059          * @param {Number} columnIndex
55060          * @param {Roo.EventObject} e
55061          */
55062         "cellclick" : true,
55063         /**
55064          * @event celldblclick
55065          * Fires when a cell is double clicked
55066          * @param {Grid} this
55067          * @param {Number} rowIndex
55068          * @param {Number} columnIndex
55069          * @param {Roo.EventObject} e
55070          */
55071         "celldblclick" : true,
55072         /**
55073          * @event rowclick
55074          * Fires when a row is clicked
55075          * @param {Grid} this
55076          * @param {Number} rowIndex
55077          * @param {Roo.EventObject} e
55078          */
55079         "rowclick" : true,
55080         /**
55081          * @event rowdblclick
55082          * Fires when a row is double clicked
55083          * @param {Grid} this
55084          * @param {Number} rowIndex
55085          * @param {Roo.EventObject} e
55086          */
55087         "rowdblclick" : true,
55088         /**
55089          * @event headerclick
55090          * Fires when a header is clicked
55091          * @param {Grid} this
55092          * @param {Number} columnIndex
55093          * @param {Roo.EventObject} e
55094          */
55095         "headerclick" : true,
55096         /**
55097          * @event headerdblclick
55098          * Fires when a header cell is double clicked
55099          * @param {Grid} this
55100          * @param {Number} columnIndex
55101          * @param {Roo.EventObject} e
55102          */
55103         "headerdblclick" : true,
55104         /**
55105          * @event rowcontextmenu
55106          * Fires when a row is right clicked
55107          * @param {Grid} this
55108          * @param {Number} rowIndex
55109          * @param {Roo.EventObject} e
55110          */
55111         "rowcontextmenu" : true,
55112         /**
55113          * @event cellcontextmenu
55114          * Fires when a cell is right clicked
55115          * @param {Grid} this
55116          * @param {Number} rowIndex
55117          * @param {Number} cellIndex
55118          * @param {Roo.EventObject} e
55119          */
55120          "cellcontextmenu" : true,
55121         /**
55122          * @event headercontextmenu
55123          * Fires when a header is right clicked
55124          * @param {Grid} this
55125          * @param {Number} columnIndex
55126          * @param {Roo.EventObject} e
55127          */
55128         "headercontextmenu" : true,
55129         /**
55130          * @event bodyscroll
55131          * Fires when the body element is scrolled
55132          * @param {Number} scrollLeft
55133          * @param {Number} scrollTop
55134          */
55135         "bodyscroll" : true,
55136         /**
55137          * @event columnresize
55138          * Fires when the user resizes a column
55139          * @param {Number} columnIndex
55140          * @param {Number} newSize
55141          */
55142         "columnresize" : true,
55143         /**
55144          * @event columnmove
55145          * Fires when the user moves a column
55146          * @param {Number} oldIndex
55147          * @param {Number} newIndex
55148          */
55149         "columnmove" : true,
55150         /**
55151          * @event startdrag
55152          * Fires when row(s) start being dragged
55153          * @param {Grid} this
55154          * @param {Roo.GridDD} dd The drag drop object
55155          * @param {event} e The raw browser event
55156          */
55157         "startdrag" : true,
55158         /**
55159          * @event enddrag
55160          * Fires when a drag operation is complete
55161          * @param {Grid} this
55162          * @param {Roo.GridDD} dd The drag drop object
55163          * @param {event} e The raw browser event
55164          */
55165         "enddrag" : true,
55166         /**
55167          * @event dragdrop
55168          * Fires when dragged row(s) are dropped on a valid DD target
55169          * @param {Grid} this
55170          * @param {Roo.GridDD} dd The drag drop object
55171          * @param {String} targetId The target drag drop object
55172          * @param {event} e The raw browser event
55173          */
55174         "dragdrop" : true,
55175         /**
55176          * @event dragover
55177          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55178          * @param {Grid} this
55179          * @param {Roo.GridDD} dd The drag drop object
55180          * @param {String} targetId The target drag drop object
55181          * @param {event} e The raw browser event
55182          */
55183         "dragover" : true,
55184         /**
55185          * @event dragenter
55186          *  Fires when the dragged row(s) first cross another DD target while being dragged
55187          * @param {Grid} this
55188          * @param {Roo.GridDD} dd The drag drop object
55189          * @param {String} targetId The target drag drop object
55190          * @param {event} e The raw browser event
55191          */
55192         "dragenter" : true,
55193         /**
55194          * @event dragout
55195          * Fires when the dragged row(s) leave another DD target while being dragged
55196          * @param {Grid} this
55197          * @param {Roo.GridDD} dd The drag drop object
55198          * @param {String} targetId The target drag drop object
55199          * @param {event} e The raw browser event
55200          */
55201         "dragout" : true,
55202         /**
55203          * @event rowclass
55204          * Fires when a row is rendered, so you can change add a style to it.
55205          * @param {GridView} gridview   The grid view
55206          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55207          */
55208         'rowclass' : true,
55209
55210         /**
55211          * @event render
55212          * Fires when the grid is rendered
55213          * @param {Grid} grid
55214          */
55215         'render' : true
55216     });
55217
55218     Roo.grid.Grid.superclass.constructor.call(this);
55219 };
55220 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55221     
55222     /**
55223      * @cfg {String} ddGroup - drag drop group.
55224      */
55225
55226     /**
55227      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55228      */
55229     minColumnWidth : 25,
55230
55231     /**
55232      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55233      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55234      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55235      */
55236     autoSizeColumns : false,
55237
55238     /**
55239      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55240      */
55241     autoSizeHeaders : true,
55242
55243     /**
55244      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55245      */
55246     monitorWindowResize : true,
55247
55248     /**
55249      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55250      * rows measured to get a columns size. Default is 0 (all rows).
55251      */
55252     maxRowsToMeasure : 0,
55253
55254     /**
55255      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55256      */
55257     trackMouseOver : true,
55258
55259     /**
55260     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55261     */
55262     
55263     /**
55264     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55265     */
55266     enableDragDrop : false,
55267     
55268     /**
55269     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55270     */
55271     enableColumnMove : true,
55272     
55273     /**
55274     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55275     */
55276     enableColumnHide : true,
55277     
55278     /**
55279     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55280     */
55281     enableRowHeightSync : false,
55282     
55283     /**
55284     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55285     */
55286     stripeRows : true,
55287     
55288     /**
55289     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55290     */
55291     autoHeight : false,
55292
55293     /**
55294      * @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.
55295      */
55296     autoExpandColumn : false,
55297
55298     /**
55299     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55300     * Default is 50.
55301     */
55302     autoExpandMin : 50,
55303
55304     /**
55305     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55306     */
55307     autoExpandMax : 1000,
55308
55309     /**
55310     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55311     */
55312     view : null,
55313
55314     /**
55315     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55316     */
55317     loadMask : false,
55318     /**
55319     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55320     */
55321     dropTarget: false,
55322     
55323    
55324     
55325     // private
55326     rendered : false,
55327
55328     /**
55329     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55330     * of a fixed width. Default is false.
55331     */
55332     /**
55333     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55334     */
55335     /**
55336      * Called once after all setup has been completed and the grid is ready to be rendered.
55337      * @return {Roo.grid.Grid} this
55338      */
55339     render : function()
55340     {
55341         var c = this.container;
55342         // try to detect autoHeight/width mode
55343         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55344             this.autoHeight = true;
55345         }
55346         var view = this.getView();
55347         view.init(this);
55348
55349         c.on("click", this.onClick, this);
55350         c.on("dblclick", this.onDblClick, this);
55351         c.on("contextmenu", this.onContextMenu, this);
55352         c.on("keydown", this.onKeyDown, this);
55353         if (Roo.isTouch) {
55354             c.on("touchstart", this.onTouchStart, this);
55355         }
55356
55357         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55358
55359         this.getSelectionModel().init(this);
55360
55361         view.render();
55362
55363         if(this.loadMask){
55364             this.loadMask = new Roo.LoadMask(this.container,
55365                     Roo.apply({store:this.dataSource}, this.loadMask));
55366         }
55367         
55368         
55369         if (this.toolbar && this.toolbar.xtype) {
55370             this.toolbar.container = this.getView().getHeaderPanel(true);
55371             this.toolbar = new Roo.Toolbar(this.toolbar);
55372         }
55373         if (this.footer && this.footer.xtype) {
55374             this.footer.dataSource = this.getDataSource();
55375             this.footer.container = this.getView().getFooterPanel(true);
55376             this.footer = Roo.factory(this.footer, Roo);
55377         }
55378         if (this.dropTarget && this.dropTarget.xtype) {
55379             delete this.dropTarget.xtype;
55380             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55381         }
55382         
55383         
55384         this.rendered = true;
55385         this.fireEvent('render', this);
55386         return this;
55387     },
55388
55389         /**
55390          * Reconfigures the grid to use a different Store and Column Model.
55391          * The View will be bound to the new objects and refreshed.
55392          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55393          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55394          */
55395     reconfigure : function(dataSource, colModel){
55396         if(this.loadMask){
55397             this.loadMask.destroy();
55398             this.loadMask = new Roo.LoadMask(this.container,
55399                     Roo.apply({store:dataSource}, this.loadMask));
55400         }
55401         this.view.bind(dataSource, colModel);
55402         this.dataSource = dataSource;
55403         this.colModel = colModel;
55404         this.view.refresh(true);
55405     },
55406
55407     // private
55408     onKeyDown : function(e){
55409         this.fireEvent("keydown", e);
55410     },
55411
55412     /**
55413      * Destroy this grid.
55414      * @param {Boolean} removeEl True to remove the element
55415      */
55416     destroy : function(removeEl, keepListeners){
55417         if(this.loadMask){
55418             this.loadMask.destroy();
55419         }
55420         var c = this.container;
55421         c.removeAllListeners();
55422         this.view.destroy();
55423         this.colModel.purgeListeners();
55424         if(!keepListeners){
55425             this.purgeListeners();
55426         }
55427         c.update("");
55428         if(removeEl === true){
55429             c.remove();
55430         }
55431     },
55432
55433     // private
55434     processEvent : function(name, e){
55435         // does this fire select???
55436         //Roo.log('grid:processEvent '  + name);
55437         
55438         if (name != 'touchstart' ) {
55439             this.fireEvent(name, e);    
55440         }
55441         
55442         var t = e.getTarget();
55443         var v = this.view;
55444         var header = v.findHeaderIndex(t);
55445         if(header !== false){
55446             var ename = name == 'touchstart' ? 'click' : name;
55447              
55448             this.fireEvent("header" + ename, this, header, e);
55449         }else{
55450             var row = v.findRowIndex(t);
55451             var cell = v.findCellIndex(t);
55452             if (name == 'touchstart') {
55453                 // first touch is always a click.
55454                 // hopefull this happens after selection is updated.?
55455                 name = false;
55456                 
55457                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55458                     var cs = this.selModel.getSelectedCell();
55459                     if (row == cs[0] && cell == cs[1]){
55460                         name = 'dblclick';
55461                     }
55462                 }
55463                 if (typeof(this.selModel.getSelections) != 'undefined') {
55464                     var cs = this.selModel.getSelections();
55465                     var ds = this.dataSource;
55466                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55467                         name = 'dblclick';
55468                     }
55469                 }
55470                 if (!name) {
55471                     return;
55472                 }
55473             }
55474             
55475             
55476             if(row !== false){
55477                 this.fireEvent("row" + name, this, row, e);
55478                 if(cell !== false){
55479                     this.fireEvent("cell" + name, this, row, cell, e);
55480                 }
55481             }
55482         }
55483     },
55484
55485     // private
55486     onClick : function(e){
55487         this.processEvent("click", e);
55488     },
55489    // private
55490     onTouchStart : function(e){
55491         this.processEvent("touchstart", e);
55492     },
55493
55494     // private
55495     onContextMenu : function(e, t){
55496         this.processEvent("contextmenu", e);
55497     },
55498
55499     // private
55500     onDblClick : function(e){
55501         this.processEvent("dblclick", e);
55502     },
55503
55504     // private
55505     walkCells : function(row, col, step, fn, scope){
55506         var cm = this.colModel, clen = cm.getColumnCount();
55507         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55508         if(step < 0){
55509             if(col < 0){
55510                 row--;
55511                 first = false;
55512             }
55513             while(row >= 0){
55514                 if(!first){
55515                     col = clen-1;
55516                 }
55517                 first = false;
55518                 while(col >= 0){
55519                     if(fn.call(scope || this, row, col, cm) === true){
55520                         return [row, col];
55521                     }
55522                     col--;
55523                 }
55524                 row--;
55525             }
55526         } else {
55527             if(col >= clen){
55528                 row++;
55529                 first = false;
55530             }
55531             while(row < rlen){
55532                 if(!first){
55533                     col = 0;
55534                 }
55535                 first = false;
55536                 while(col < clen){
55537                     if(fn.call(scope || this, row, col, cm) === true){
55538                         return [row, col];
55539                     }
55540                     col++;
55541                 }
55542                 row++;
55543             }
55544         }
55545         return null;
55546     },
55547
55548     // private
55549     getSelections : function(){
55550         return this.selModel.getSelections();
55551     },
55552
55553     /**
55554      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55555      * but if manual update is required this method will initiate it.
55556      */
55557     autoSize : function(){
55558         if(this.rendered){
55559             this.view.layout();
55560             if(this.view.adjustForScroll){
55561                 this.view.adjustForScroll();
55562             }
55563         }
55564     },
55565
55566     /**
55567      * Returns the grid's underlying element.
55568      * @return {Element} The element
55569      */
55570     getGridEl : function(){
55571         return this.container;
55572     },
55573
55574     // private for compatibility, overridden by editor grid
55575     stopEditing : function(){},
55576
55577     /**
55578      * Returns the grid's SelectionModel.
55579      * @return {SelectionModel}
55580      */
55581     getSelectionModel : function(){
55582         if(!this.selModel){
55583             this.selModel = new Roo.grid.RowSelectionModel();
55584         }
55585         return this.selModel;
55586     },
55587
55588     /**
55589      * Returns the grid's DataSource.
55590      * @return {DataSource}
55591      */
55592     getDataSource : function(){
55593         return this.dataSource;
55594     },
55595
55596     /**
55597      * Returns the grid's ColumnModel.
55598      * @return {ColumnModel}
55599      */
55600     getColumnModel : function(){
55601         return this.colModel;
55602     },
55603
55604     /**
55605      * Returns the grid's GridView object.
55606      * @return {GridView}
55607      */
55608     getView : function(){
55609         if(!this.view){
55610             this.view = new Roo.grid.GridView(this.viewConfig);
55611         }
55612         return this.view;
55613     },
55614     /**
55615      * Called to get grid's drag proxy text, by default returns this.ddText.
55616      * @return {String}
55617      */
55618     getDragDropText : function(){
55619         var count = this.selModel.getCount();
55620         return String.format(this.ddText, count, count == 1 ? '' : 's');
55621     }
55622 });
55623 /**
55624  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55625  * %0 is replaced with the number of selected rows.
55626  * @type String
55627  */
55628 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55629  * Based on:
55630  * Ext JS Library 1.1.1
55631  * Copyright(c) 2006-2007, Ext JS, LLC.
55632  *
55633  * Originally Released Under LGPL - original licence link has changed is not relivant.
55634  *
55635  * Fork - LGPL
55636  * <script type="text/javascript">
55637  */
55638  
55639 Roo.grid.AbstractGridView = function(){
55640         this.grid = null;
55641         
55642         this.events = {
55643             "beforerowremoved" : true,
55644             "beforerowsinserted" : true,
55645             "beforerefresh" : true,
55646             "rowremoved" : true,
55647             "rowsinserted" : true,
55648             "rowupdated" : true,
55649             "refresh" : true
55650         };
55651     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55652 };
55653
55654 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55655     rowClass : "x-grid-row",
55656     cellClass : "x-grid-cell",
55657     tdClass : "x-grid-td",
55658     hdClass : "x-grid-hd",
55659     splitClass : "x-grid-hd-split",
55660     
55661     init: function(grid){
55662         this.grid = grid;
55663                 var cid = this.grid.getGridEl().id;
55664         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55665         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55666         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55667         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55668         },
55669         
55670     getColumnRenderers : function(){
55671         var renderers = [];
55672         var cm = this.grid.colModel;
55673         var colCount = cm.getColumnCount();
55674         for(var i = 0; i < colCount; i++){
55675             renderers[i] = cm.getRenderer(i);
55676         }
55677         return renderers;
55678     },
55679     
55680     getColumnIds : function(){
55681         var ids = [];
55682         var cm = this.grid.colModel;
55683         var colCount = cm.getColumnCount();
55684         for(var i = 0; i < colCount; i++){
55685             ids[i] = cm.getColumnId(i);
55686         }
55687         return ids;
55688     },
55689     
55690     getDataIndexes : function(){
55691         if(!this.indexMap){
55692             this.indexMap = this.buildIndexMap();
55693         }
55694         return this.indexMap.colToData;
55695     },
55696     
55697     getColumnIndexByDataIndex : function(dataIndex){
55698         if(!this.indexMap){
55699             this.indexMap = this.buildIndexMap();
55700         }
55701         return this.indexMap.dataToCol[dataIndex];
55702     },
55703     
55704     /**
55705      * Set a css style for a column dynamically. 
55706      * @param {Number} colIndex The index of the column
55707      * @param {String} name The css property name
55708      * @param {String} value The css value
55709      */
55710     setCSSStyle : function(colIndex, name, value){
55711         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55712         Roo.util.CSS.updateRule(selector, name, value);
55713     },
55714     
55715     generateRules : function(cm){
55716         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55717         Roo.util.CSS.removeStyleSheet(rulesId);
55718         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55719             var cid = cm.getColumnId(i);
55720             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55721                          this.tdSelector, cid, " {\n}\n",
55722                          this.hdSelector, cid, " {\n}\n",
55723                          this.splitSelector, cid, " {\n}\n");
55724         }
55725         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55726     }
55727 });/*
55728  * Based on:
55729  * Ext JS Library 1.1.1
55730  * Copyright(c) 2006-2007, Ext JS, LLC.
55731  *
55732  * Originally Released Under LGPL - original licence link has changed is not relivant.
55733  *
55734  * Fork - LGPL
55735  * <script type="text/javascript">
55736  */
55737
55738 // private
55739 // This is a support class used internally by the Grid components
55740 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55741     this.grid = grid;
55742     this.view = grid.getView();
55743     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55744     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55745     if(hd2){
55746         this.setHandleElId(Roo.id(hd));
55747         this.setOuterHandleElId(Roo.id(hd2));
55748     }
55749     this.scroll = false;
55750 };
55751 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55752     maxDragWidth: 120,
55753     getDragData : function(e){
55754         var t = Roo.lib.Event.getTarget(e);
55755         var h = this.view.findHeaderCell(t);
55756         if(h){
55757             return {ddel: h.firstChild, header:h};
55758         }
55759         return false;
55760     },
55761
55762     onInitDrag : function(e){
55763         this.view.headersDisabled = true;
55764         var clone = this.dragData.ddel.cloneNode(true);
55765         clone.id = Roo.id();
55766         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55767         this.proxy.update(clone);
55768         return true;
55769     },
55770
55771     afterValidDrop : function(){
55772         var v = this.view;
55773         setTimeout(function(){
55774             v.headersDisabled = false;
55775         }, 50);
55776     },
55777
55778     afterInvalidDrop : function(){
55779         var v = this.view;
55780         setTimeout(function(){
55781             v.headersDisabled = false;
55782         }, 50);
55783     }
55784 });
55785 /*
55786  * Based on:
55787  * Ext JS Library 1.1.1
55788  * Copyright(c) 2006-2007, Ext JS, LLC.
55789  *
55790  * Originally Released Under LGPL - original licence link has changed is not relivant.
55791  *
55792  * Fork - LGPL
55793  * <script type="text/javascript">
55794  */
55795 // private
55796 // This is a support class used internally by the Grid components
55797 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55798     this.grid = grid;
55799     this.view = grid.getView();
55800     // split the proxies so they don't interfere with mouse events
55801     this.proxyTop = Roo.DomHelper.append(document.body, {
55802         cls:"col-move-top", html:"&#160;"
55803     }, true);
55804     this.proxyBottom = Roo.DomHelper.append(document.body, {
55805         cls:"col-move-bottom", html:"&#160;"
55806     }, true);
55807     this.proxyTop.hide = this.proxyBottom.hide = function(){
55808         this.setLeftTop(-100,-100);
55809         this.setStyle("visibility", "hidden");
55810     };
55811     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55812     // temporarily disabled
55813     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55814     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55815 };
55816 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55817     proxyOffsets : [-4, -9],
55818     fly: Roo.Element.fly,
55819
55820     getTargetFromEvent : function(e){
55821         var t = Roo.lib.Event.getTarget(e);
55822         var cindex = this.view.findCellIndex(t);
55823         if(cindex !== false){
55824             return this.view.getHeaderCell(cindex);
55825         }
55826         return null;
55827     },
55828
55829     nextVisible : function(h){
55830         var v = this.view, cm = this.grid.colModel;
55831         h = h.nextSibling;
55832         while(h){
55833             if(!cm.isHidden(v.getCellIndex(h))){
55834                 return h;
55835             }
55836             h = h.nextSibling;
55837         }
55838         return null;
55839     },
55840
55841     prevVisible : function(h){
55842         var v = this.view, cm = this.grid.colModel;
55843         h = h.prevSibling;
55844         while(h){
55845             if(!cm.isHidden(v.getCellIndex(h))){
55846                 return h;
55847             }
55848             h = h.prevSibling;
55849         }
55850         return null;
55851     },
55852
55853     positionIndicator : function(h, n, e){
55854         var x = Roo.lib.Event.getPageX(e);
55855         var r = Roo.lib.Dom.getRegion(n.firstChild);
55856         var px, pt, py = r.top + this.proxyOffsets[1];
55857         if((r.right - x) <= (r.right-r.left)/2){
55858             px = r.right+this.view.borderWidth;
55859             pt = "after";
55860         }else{
55861             px = r.left;
55862             pt = "before";
55863         }
55864         var oldIndex = this.view.getCellIndex(h);
55865         var newIndex = this.view.getCellIndex(n);
55866
55867         if(this.grid.colModel.isFixed(newIndex)){
55868             return false;
55869         }
55870
55871         var locked = this.grid.colModel.isLocked(newIndex);
55872
55873         if(pt == "after"){
55874             newIndex++;
55875         }
55876         if(oldIndex < newIndex){
55877             newIndex--;
55878         }
55879         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55880             return false;
55881         }
55882         px +=  this.proxyOffsets[0];
55883         this.proxyTop.setLeftTop(px, py);
55884         this.proxyTop.show();
55885         if(!this.bottomOffset){
55886             this.bottomOffset = this.view.mainHd.getHeight();
55887         }
55888         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55889         this.proxyBottom.show();
55890         return pt;
55891     },
55892
55893     onNodeEnter : function(n, dd, e, data){
55894         if(data.header != n){
55895             this.positionIndicator(data.header, n, e);
55896         }
55897     },
55898
55899     onNodeOver : function(n, dd, e, data){
55900         var result = false;
55901         if(data.header != n){
55902             result = this.positionIndicator(data.header, n, e);
55903         }
55904         if(!result){
55905             this.proxyTop.hide();
55906             this.proxyBottom.hide();
55907         }
55908         return result ? this.dropAllowed : this.dropNotAllowed;
55909     },
55910
55911     onNodeOut : function(n, dd, e, data){
55912         this.proxyTop.hide();
55913         this.proxyBottom.hide();
55914     },
55915
55916     onNodeDrop : function(n, dd, e, data){
55917         var h = data.header;
55918         if(h != n){
55919             var cm = this.grid.colModel;
55920             var x = Roo.lib.Event.getPageX(e);
55921             var r = Roo.lib.Dom.getRegion(n.firstChild);
55922             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55923             var oldIndex = this.view.getCellIndex(h);
55924             var newIndex = this.view.getCellIndex(n);
55925             var locked = cm.isLocked(newIndex);
55926             if(pt == "after"){
55927                 newIndex++;
55928             }
55929             if(oldIndex < newIndex){
55930                 newIndex--;
55931             }
55932             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55933                 return false;
55934             }
55935             cm.setLocked(oldIndex, locked, true);
55936             cm.moveColumn(oldIndex, newIndex);
55937             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55938             return true;
55939         }
55940         return false;
55941     }
55942 });
55943 /*
55944  * Based on:
55945  * Ext JS Library 1.1.1
55946  * Copyright(c) 2006-2007, Ext JS, LLC.
55947  *
55948  * Originally Released Under LGPL - original licence link has changed is not relivant.
55949  *
55950  * Fork - LGPL
55951  * <script type="text/javascript">
55952  */
55953   
55954 /**
55955  * @class Roo.grid.GridView
55956  * @extends Roo.util.Observable
55957  *
55958  * @constructor
55959  * @param {Object} config
55960  */
55961 Roo.grid.GridView = function(config){
55962     Roo.grid.GridView.superclass.constructor.call(this);
55963     this.el = null;
55964
55965     Roo.apply(this, config);
55966 };
55967
55968 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55969
55970     unselectable :  'unselectable="on"',
55971     unselectableCls :  'x-unselectable',
55972     
55973     
55974     rowClass : "x-grid-row",
55975
55976     cellClass : "x-grid-col",
55977
55978     tdClass : "x-grid-td",
55979
55980     hdClass : "x-grid-hd",
55981
55982     splitClass : "x-grid-split",
55983
55984     sortClasses : ["sort-asc", "sort-desc"],
55985
55986     enableMoveAnim : false,
55987
55988     hlColor: "C3DAF9",
55989
55990     dh : Roo.DomHelper,
55991
55992     fly : Roo.Element.fly,
55993
55994     css : Roo.util.CSS,
55995
55996     borderWidth: 1,
55997
55998     splitOffset: 3,
55999
56000     scrollIncrement : 22,
56001
56002     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
56003
56004     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
56005
56006     bind : function(ds, cm){
56007         if(this.ds){
56008             this.ds.un("load", this.onLoad, this);
56009             this.ds.un("datachanged", this.onDataChange, this);
56010             this.ds.un("add", this.onAdd, this);
56011             this.ds.un("remove", this.onRemove, this);
56012             this.ds.un("update", this.onUpdate, this);
56013             this.ds.un("clear", this.onClear, this);
56014         }
56015         if(ds){
56016             ds.on("load", this.onLoad, this);
56017             ds.on("datachanged", this.onDataChange, this);
56018             ds.on("add", this.onAdd, this);
56019             ds.on("remove", this.onRemove, this);
56020             ds.on("update", this.onUpdate, this);
56021             ds.on("clear", this.onClear, this);
56022         }
56023         this.ds = ds;
56024
56025         if(this.cm){
56026             this.cm.un("widthchange", this.onColWidthChange, this);
56027             this.cm.un("headerchange", this.onHeaderChange, this);
56028             this.cm.un("hiddenchange", this.onHiddenChange, this);
56029             this.cm.un("columnmoved", this.onColumnMove, this);
56030             this.cm.un("columnlockchange", this.onColumnLock, this);
56031         }
56032         if(cm){
56033             this.generateRules(cm);
56034             cm.on("widthchange", this.onColWidthChange, this);
56035             cm.on("headerchange", this.onHeaderChange, this);
56036             cm.on("hiddenchange", this.onHiddenChange, this);
56037             cm.on("columnmoved", this.onColumnMove, this);
56038             cm.on("columnlockchange", this.onColumnLock, this);
56039         }
56040         this.cm = cm;
56041     },
56042
56043     init: function(grid){
56044         Roo.grid.GridView.superclass.init.call(this, grid);
56045
56046         this.bind(grid.dataSource, grid.colModel);
56047
56048         grid.on("headerclick", this.handleHeaderClick, this);
56049
56050         if(grid.trackMouseOver){
56051             grid.on("mouseover", this.onRowOver, this);
56052             grid.on("mouseout", this.onRowOut, this);
56053         }
56054         grid.cancelTextSelection = function(){};
56055         this.gridId = grid.id;
56056
56057         var tpls = this.templates || {};
56058
56059         if(!tpls.master){
56060             tpls.master = new Roo.Template(
56061                '<div class="x-grid" hidefocus="true">',
56062                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56063                   '<div class="x-grid-topbar"></div>',
56064                   '<div class="x-grid-scroller"><div></div></div>',
56065                   '<div class="x-grid-locked">',
56066                       '<div class="x-grid-header">{lockedHeader}</div>',
56067                       '<div class="x-grid-body">{lockedBody}</div>',
56068                   "</div>",
56069                   '<div class="x-grid-viewport">',
56070                       '<div class="x-grid-header">{header}</div>',
56071                       '<div class="x-grid-body">{body}</div>',
56072                   "</div>",
56073                   '<div class="x-grid-bottombar"></div>',
56074                  
56075                   '<div class="x-grid-resize-proxy">&#160;</div>',
56076                "</div>"
56077             );
56078             tpls.master.disableformats = true;
56079         }
56080
56081         if(!tpls.header){
56082             tpls.header = new Roo.Template(
56083                '<table border="0" cellspacing="0" cellpadding="0">',
56084                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56085                "</table>{splits}"
56086             );
56087             tpls.header.disableformats = true;
56088         }
56089         tpls.header.compile();
56090
56091         if(!tpls.hcell){
56092             tpls.hcell = new Roo.Template(
56093                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56094                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56095                 "</div></td>"
56096              );
56097              tpls.hcell.disableFormats = true;
56098         }
56099         tpls.hcell.compile();
56100
56101         if(!tpls.hsplit){
56102             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56103                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56104             tpls.hsplit.disableFormats = true;
56105         }
56106         tpls.hsplit.compile();
56107
56108         if(!tpls.body){
56109             tpls.body = new Roo.Template(
56110                '<table border="0" cellspacing="0" cellpadding="0">',
56111                "<tbody>{rows}</tbody>",
56112                "</table>"
56113             );
56114             tpls.body.disableFormats = true;
56115         }
56116         tpls.body.compile();
56117
56118         if(!tpls.row){
56119             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56120             tpls.row.disableFormats = true;
56121         }
56122         tpls.row.compile();
56123
56124         if(!tpls.cell){
56125             tpls.cell = new Roo.Template(
56126                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56127                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56128                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56129                 "</td>"
56130             );
56131             tpls.cell.disableFormats = true;
56132         }
56133         tpls.cell.compile();
56134
56135         this.templates = tpls;
56136     },
56137
56138     // remap these for backwards compat
56139     onColWidthChange : function(){
56140         this.updateColumns.apply(this, arguments);
56141     },
56142     onHeaderChange : function(){
56143         this.updateHeaders.apply(this, arguments);
56144     }, 
56145     onHiddenChange : function(){
56146         this.handleHiddenChange.apply(this, arguments);
56147     },
56148     onColumnMove : function(){
56149         this.handleColumnMove.apply(this, arguments);
56150     },
56151     onColumnLock : function(){
56152         this.handleLockChange.apply(this, arguments);
56153     },
56154
56155     onDataChange : function(){
56156         this.refresh();
56157         this.updateHeaderSortState();
56158     },
56159
56160     onClear : function(){
56161         this.refresh();
56162     },
56163
56164     onUpdate : function(ds, record){
56165         this.refreshRow(record);
56166     },
56167
56168     refreshRow : function(record){
56169         var ds = this.ds, index;
56170         if(typeof record == 'number'){
56171             index = record;
56172             record = ds.getAt(index);
56173         }else{
56174             index = ds.indexOf(record);
56175         }
56176         this.insertRows(ds, index, index, true);
56177         this.onRemove(ds, record, index+1, true);
56178         this.syncRowHeights(index, index);
56179         this.layout();
56180         this.fireEvent("rowupdated", this, index, record);
56181     },
56182
56183     onAdd : function(ds, records, index){
56184         this.insertRows(ds, index, index + (records.length-1));
56185     },
56186
56187     onRemove : function(ds, record, index, isUpdate){
56188         if(isUpdate !== true){
56189             this.fireEvent("beforerowremoved", this, index, record);
56190         }
56191         var bt = this.getBodyTable(), lt = this.getLockedTable();
56192         if(bt.rows[index]){
56193             bt.firstChild.removeChild(bt.rows[index]);
56194         }
56195         if(lt.rows[index]){
56196             lt.firstChild.removeChild(lt.rows[index]);
56197         }
56198         if(isUpdate !== true){
56199             this.stripeRows(index);
56200             this.syncRowHeights(index, index);
56201             this.layout();
56202             this.fireEvent("rowremoved", this, index, record);
56203         }
56204     },
56205
56206     onLoad : function(){
56207         this.scrollToTop();
56208     },
56209
56210     /**
56211      * Scrolls the grid to the top
56212      */
56213     scrollToTop : function(){
56214         if(this.scroller){
56215             this.scroller.dom.scrollTop = 0;
56216             this.syncScroll();
56217         }
56218     },
56219
56220     /**
56221      * Gets a panel in the header of the grid that can be used for toolbars etc.
56222      * After modifying the contents of this panel a call to grid.autoSize() may be
56223      * required to register any changes in size.
56224      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56225      * @return Roo.Element
56226      */
56227     getHeaderPanel : function(doShow){
56228         if(doShow){
56229             this.headerPanel.show();
56230         }
56231         return this.headerPanel;
56232     },
56233
56234     /**
56235      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56236      * After modifying the contents of this panel a call to grid.autoSize() may be
56237      * required to register any changes in size.
56238      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56239      * @return Roo.Element
56240      */
56241     getFooterPanel : function(doShow){
56242         if(doShow){
56243             this.footerPanel.show();
56244         }
56245         return this.footerPanel;
56246     },
56247
56248     initElements : function(){
56249         var E = Roo.Element;
56250         var el = this.grid.getGridEl().dom.firstChild;
56251         var cs = el.childNodes;
56252
56253         this.el = new E(el);
56254         
56255          this.focusEl = new E(el.firstChild);
56256         this.focusEl.swallowEvent("click", true);
56257         
56258         this.headerPanel = new E(cs[1]);
56259         this.headerPanel.enableDisplayMode("block");
56260
56261         this.scroller = new E(cs[2]);
56262         this.scrollSizer = new E(this.scroller.dom.firstChild);
56263
56264         this.lockedWrap = new E(cs[3]);
56265         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56266         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56267
56268         this.mainWrap = new E(cs[4]);
56269         this.mainHd = new E(this.mainWrap.dom.firstChild);
56270         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56271
56272         this.footerPanel = new E(cs[5]);
56273         this.footerPanel.enableDisplayMode("block");
56274
56275         this.resizeProxy = new E(cs[6]);
56276
56277         this.headerSelector = String.format(
56278            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56279            this.lockedHd.id, this.mainHd.id
56280         );
56281
56282         this.splitterSelector = String.format(
56283            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56284            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56285         );
56286     },
56287     idToCssName : function(s)
56288     {
56289         return s.replace(/[^a-z0-9]+/ig, '-');
56290     },
56291
56292     getHeaderCell : function(index){
56293         return Roo.DomQuery.select(this.headerSelector)[index];
56294     },
56295
56296     getHeaderCellMeasure : function(index){
56297         return this.getHeaderCell(index).firstChild;
56298     },
56299
56300     getHeaderCellText : function(index){
56301         return this.getHeaderCell(index).firstChild.firstChild;
56302     },
56303
56304     getLockedTable : function(){
56305         return this.lockedBody.dom.firstChild;
56306     },
56307
56308     getBodyTable : function(){
56309         return this.mainBody.dom.firstChild;
56310     },
56311
56312     getLockedRow : function(index){
56313         return this.getLockedTable().rows[index];
56314     },
56315
56316     getRow : function(index){
56317         return this.getBodyTable().rows[index];
56318     },
56319
56320     getRowComposite : function(index){
56321         if(!this.rowEl){
56322             this.rowEl = new Roo.CompositeElementLite();
56323         }
56324         var els = [], lrow, mrow;
56325         if(lrow = this.getLockedRow(index)){
56326             els.push(lrow);
56327         }
56328         if(mrow = this.getRow(index)){
56329             els.push(mrow);
56330         }
56331         this.rowEl.elements = els;
56332         return this.rowEl;
56333     },
56334     /**
56335      * Gets the 'td' of the cell
56336      * 
56337      * @param {Integer} rowIndex row to select
56338      * @param {Integer} colIndex column to select
56339      * 
56340      * @return {Object} 
56341      */
56342     getCell : function(rowIndex, colIndex){
56343         var locked = this.cm.getLockedCount();
56344         var source;
56345         if(colIndex < locked){
56346             source = this.lockedBody.dom.firstChild;
56347         }else{
56348             source = this.mainBody.dom.firstChild;
56349             colIndex -= locked;
56350         }
56351         return source.rows[rowIndex].childNodes[colIndex];
56352     },
56353
56354     getCellText : function(rowIndex, colIndex){
56355         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56356     },
56357
56358     getCellBox : function(cell){
56359         var b = this.fly(cell).getBox();
56360         if(Roo.isOpera){ // opera fails to report the Y
56361             b.y = cell.offsetTop + this.mainBody.getY();
56362         }
56363         return b;
56364     },
56365
56366     getCellIndex : function(cell){
56367         var id = String(cell.className).match(this.cellRE);
56368         if(id){
56369             return parseInt(id[1], 10);
56370         }
56371         return 0;
56372     },
56373
56374     findHeaderIndex : function(n){
56375         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56376         return r ? this.getCellIndex(r) : false;
56377     },
56378
56379     findHeaderCell : function(n){
56380         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56381         return r ? r : false;
56382     },
56383
56384     findRowIndex : function(n){
56385         if(!n){
56386             return false;
56387         }
56388         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56389         return r ? r.rowIndex : false;
56390     },
56391
56392     findCellIndex : function(node){
56393         var stop = this.el.dom;
56394         while(node && node != stop){
56395             if(this.findRE.test(node.className)){
56396                 return this.getCellIndex(node);
56397             }
56398             node = node.parentNode;
56399         }
56400         return false;
56401     },
56402
56403     getColumnId : function(index){
56404         return this.cm.getColumnId(index);
56405     },
56406
56407     getSplitters : function()
56408     {
56409         if(this.splitterSelector){
56410            return Roo.DomQuery.select(this.splitterSelector);
56411         }else{
56412             return null;
56413       }
56414     },
56415
56416     getSplitter : function(index){
56417         return this.getSplitters()[index];
56418     },
56419
56420     onRowOver : function(e, t){
56421         var row;
56422         if((row = this.findRowIndex(t)) !== false){
56423             this.getRowComposite(row).addClass("x-grid-row-over");
56424         }
56425     },
56426
56427     onRowOut : function(e, t){
56428         var row;
56429         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56430             this.getRowComposite(row).removeClass("x-grid-row-over");
56431         }
56432     },
56433
56434     renderHeaders : function(){
56435         var cm = this.cm;
56436         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56437         var cb = [], lb = [], sb = [], lsb = [], p = {};
56438         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56439             p.cellId = "x-grid-hd-0-" + i;
56440             p.splitId = "x-grid-csplit-0-" + i;
56441             p.id = cm.getColumnId(i);
56442             p.value = cm.getColumnHeader(i) || "";
56443             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56444             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56445             if(!cm.isLocked(i)){
56446                 cb[cb.length] = ct.apply(p);
56447                 sb[sb.length] = st.apply(p);
56448             }else{
56449                 lb[lb.length] = ct.apply(p);
56450                 lsb[lsb.length] = st.apply(p);
56451             }
56452         }
56453         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56454                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56455     },
56456
56457     updateHeaders : function(){
56458         var html = this.renderHeaders();
56459         this.lockedHd.update(html[0]);
56460         this.mainHd.update(html[1]);
56461     },
56462
56463     /**
56464      * Focuses the specified row.
56465      * @param {Number} row The row index
56466      */
56467     focusRow : function(row)
56468     {
56469         //Roo.log('GridView.focusRow');
56470         var x = this.scroller.dom.scrollLeft;
56471         this.focusCell(row, 0, false);
56472         this.scroller.dom.scrollLeft = x;
56473     },
56474
56475     /**
56476      * Focuses the specified cell.
56477      * @param {Number} row The row index
56478      * @param {Number} col The column index
56479      * @param {Boolean} hscroll false to disable horizontal scrolling
56480      */
56481     focusCell : function(row, col, hscroll)
56482     {
56483         //Roo.log('GridView.focusCell');
56484         var el = this.ensureVisible(row, col, hscroll);
56485         this.focusEl.alignTo(el, "tl-tl");
56486         if(Roo.isGecko){
56487             this.focusEl.focus();
56488         }else{
56489             this.focusEl.focus.defer(1, this.focusEl);
56490         }
56491     },
56492
56493     /**
56494      * Scrolls the specified cell into view
56495      * @param {Number} row The row index
56496      * @param {Number} col The column index
56497      * @param {Boolean} hscroll false to disable horizontal scrolling
56498      */
56499     ensureVisible : function(row, col, hscroll)
56500     {
56501         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56502         //return null; //disable for testing.
56503         if(typeof row != "number"){
56504             row = row.rowIndex;
56505         }
56506         if(row < 0 && row >= this.ds.getCount()){
56507             return  null;
56508         }
56509         col = (col !== undefined ? col : 0);
56510         var cm = this.grid.colModel;
56511         while(cm.isHidden(col)){
56512             col++;
56513         }
56514
56515         var el = this.getCell(row, col);
56516         if(!el){
56517             return null;
56518         }
56519         var c = this.scroller.dom;
56520
56521         var ctop = parseInt(el.offsetTop, 10);
56522         var cleft = parseInt(el.offsetLeft, 10);
56523         var cbot = ctop + el.offsetHeight;
56524         var cright = cleft + el.offsetWidth;
56525         
56526         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56527         var stop = parseInt(c.scrollTop, 10);
56528         var sleft = parseInt(c.scrollLeft, 10);
56529         var sbot = stop + ch;
56530         var sright = sleft + c.clientWidth;
56531         /*
56532         Roo.log('GridView.ensureVisible:' +
56533                 ' ctop:' + ctop +
56534                 ' c.clientHeight:' + c.clientHeight +
56535                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56536                 ' stop:' + stop +
56537                 ' cbot:' + cbot +
56538                 ' sbot:' + sbot +
56539                 ' ch:' + ch  
56540                 );
56541         */
56542         if(ctop < stop){
56543              c.scrollTop = ctop;
56544             //Roo.log("set scrolltop to ctop DISABLE?");
56545         }else if(cbot > sbot){
56546             //Roo.log("set scrolltop to cbot-ch");
56547             c.scrollTop = cbot-ch;
56548         }
56549         
56550         if(hscroll !== false){
56551             if(cleft < sleft){
56552                 c.scrollLeft = cleft;
56553             }else if(cright > sright){
56554                 c.scrollLeft = cright-c.clientWidth;
56555             }
56556         }
56557          
56558         return el;
56559     },
56560
56561     updateColumns : function(){
56562         this.grid.stopEditing();
56563         var cm = this.grid.colModel, colIds = this.getColumnIds();
56564         //var totalWidth = cm.getTotalWidth();
56565         var pos = 0;
56566         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56567             //if(cm.isHidden(i)) continue;
56568             var w = cm.getColumnWidth(i);
56569             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56570             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56571         }
56572         this.updateSplitters();
56573     },
56574
56575     generateRules : function(cm){
56576         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56577         Roo.util.CSS.removeStyleSheet(rulesId);
56578         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56579             var cid = cm.getColumnId(i);
56580             var align = '';
56581             if(cm.config[i].align){
56582                 align = 'text-align:'+cm.config[i].align+';';
56583             }
56584             var hidden = '';
56585             if(cm.isHidden(i)){
56586                 hidden = 'display:none;';
56587             }
56588             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56589             ruleBuf.push(
56590                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56591                     this.hdSelector, cid, " {\n", align, width, "}\n",
56592                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56593                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56594         }
56595         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56596     },
56597
56598     updateSplitters : function(){
56599         var cm = this.cm, s = this.getSplitters();
56600         if(s){ // splitters not created yet
56601             var pos = 0, locked = true;
56602             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56603                 if(cm.isHidden(i)) {
56604                     continue;
56605                 }
56606                 var w = cm.getColumnWidth(i); // make sure it's a number
56607                 if(!cm.isLocked(i) && locked){
56608                     pos = 0;
56609                     locked = false;
56610                 }
56611                 pos += w;
56612                 s[i].style.left = (pos-this.splitOffset) + "px";
56613             }
56614         }
56615     },
56616
56617     handleHiddenChange : function(colModel, colIndex, hidden){
56618         if(hidden){
56619             this.hideColumn(colIndex);
56620         }else{
56621             this.unhideColumn(colIndex);
56622         }
56623     },
56624
56625     hideColumn : function(colIndex){
56626         var cid = this.getColumnId(colIndex);
56627         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56628         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56629         if(Roo.isSafari){
56630             this.updateHeaders();
56631         }
56632         this.updateSplitters();
56633         this.layout();
56634     },
56635
56636     unhideColumn : function(colIndex){
56637         var cid = this.getColumnId(colIndex);
56638         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56639         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56640
56641         if(Roo.isSafari){
56642             this.updateHeaders();
56643         }
56644         this.updateSplitters();
56645         this.layout();
56646     },
56647
56648     insertRows : function(dm, firstRow, lastRow, isUpdate){
56649         if(firstRow == 0 && lastRow == dm.getCount()-1){
56650             this.refresh();
56651         }else{
56652             if(!isUpdate){
56653                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56654             }
56655             var s = this.getScrollState();
56656             var markup = this.renderRows(firstRow, lastRow);
56657             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56658             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56659             this.restoreScroll(s);
56660             if(!isUpdate){
56661                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56662                 this.syncRowHeights(firstRow, lastRow);
56663                 this.stripeRows(firstRow);
56664                 this.layout();
56665             }
56666         }
56667     },
56668
56669     bufferRows : function(markup, target, index){
56670         var before = null, trows = target.rows, tbody = target.tBodies[0];
56671         if(index < trows.length){
56672             before = trows[index];
56673         }
56674         var b = document.createElement("div");
56675         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56676         var rows = b.firstChild.rows;
56677         for(var i = 0, len = rows.length; i < len; i++){
56678             if(before){
56679                 tbody.insertBefore(rows[0], before);
56680             }else{
56681                 tbody.appendChild(rows[0]);
56682             }
56683         }
56684         b.innerHTML = "";
56685         b = null;
56686     },
56687
56688     deleteRows : function(dm, firstRow, lastRow){
56689         if(dm.getRowCount()<1){
56690             this.fireEvent("beforerefresh", this);
56691             this.mainBody.update("");
56692             this.lockedBody.update("");
56693             this.fireEvent("refresh", this);
56694         }else{
56695             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56696             var bt = this.getBodyTable();
56697             var tbody = bt.firstChild;
56698             var rows = bt.rows;
56699             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56700                 tbody.removeChild(rows[firstRow]);
56701             }
56702             this.stripeRows(firstRow);
56703             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56704         }
56705     },
56706
56707     updateRows : function(dataSource, firstRow, lastRow){
56708         var s = this.getScrollState();
56709         this.refresh();
56710         this.restoreScroll(s);
56711     },
56712
56713     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56714         if(!noRefresh){
56715            this.refresh();
56716         }
56717         this.updateHeaderSortState();
56718     },
56719
56720     getScrollState : function(){
56721         
56722         var sb = this.scroller.dom;
56723         return {left: sb.scrollLeft, top: sb.scrollTop};
56724     },
56725
56726     stripeRows : function(startRow){
56727         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56728             return;
56729         }
56730         startRow = startRow || 0;
56731         var rows = this.getBodyTable().rows;
56732         var lrows = this.getLockedTable().rows;
56733         var cls = ' x-grid-row-alt ';
56734         for(var i = startRow, len = rows.length; i < len; i++){
56735             var row = rows[i], lrow = lrows[i];
56736             var isAlt = ((i+1) % 2 == 0);
56737             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56738             if(isAlt == hasAlt){
56739                 continue;
56740             }
56741             if(isAlt){
56742                 row.className += " x-grid-row-alt";
56743             }else{
56744                 row.className = row.className.replace("x-grid-row-alt", "");
56745             }
56746             if(lrow){
56747                 lrow.className = row.className;
56748             }
56749         }
56750     },
56751
56752     restoreScroll : function(state){
56753         //Roo.log('GridView.restoreScroll');
56754         var sb = this.scroller.dom;
56755         sb.scrollLeft = state.left;
56756         sb.scrollTop = state.top;
56757         this.syncScroll();
56758     },
56759
56760     syncScroll : function(){
56761         //Roo.log('GridView.syncScroll');
56762         var sb = this.scroller.dom;
56763         var sh = this.mainHd.dom;
56764         var bs = this.mainBody.dom;
56765         var lv = this.lockedBody.dom;
56766         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56767         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56768     },
56769
56770     handleScroll : function(e){
56771         this.syncScroll();
56772         var sb = this.scroller.dom;
56773         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56774         e.stopEvent();
56775     },
56776
56777     handleWheel : function(e){
56778         var d = e.getWheelDelta();
56779         this.scroller.dom.scrollTop -= d*22;
56780         // set this here to prevent jumpy scrolling on large tables
56781         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56782         e.stopEvent();
56783     },
56784
56785     renderRows : function(startRow, endRow){
56786         // pull in all the crap needed to render rows
56787         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56788         var colCount = cm.getColumnCount();
56789
56790         if(ds.getCount() < 1){
56791             return ["", ""];
56792         }
56793
56794         // build a map for all the columns
56795         var cs = [];
56796         for(var i = 0; i < colCount; i++){
56797             var name = cm.getDataIndex(i);
56798             cs[i] = {
56799                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56800                 renderer : cm.getRenderer(i),
56801                 id : cm.getColumnId(i),
56802                 locked : cm.isLocked(i),
56803                 has_editor : cm.isCellEditable(i)
56804             };
56805         }
56806
56807         startRow = startRow || 0;
56808         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56809
56810         // records to render
56811         var rs = ds.getRange(startRow, endRow);
56812
56813         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56814     },
56815
56816     // As much as I hate to duplicate code, this was branched because FireFox really hates
56817     // [].join("") on strings. The performance difference was substantial enough to
56818     // branch this function
56819     doRender : Roo.isGecko ?
56820             function(cs, rs, ds, startRow, colCount, stripe){
56821                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56822                 // buffers
56823                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56824                 
56825                 var hasListener = this.grid.hasListener('rowclass');
56826                 var rowcfg = {};
56827                 for(var j = 0, len = rs.length; j < len; j++){
56828                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56829                     for(var i = 0; i < colCount; i++){
56830                         c = cs[i];
56831                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56832                         p.id = c.id;
56833                         p.css = p.attr = "";
56834                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56835                         if(p.value == undefined || p.value === "") {
56836                             p.value = "&#160;";
56837                         }
56838                         if(c.has_editor){
56839                             p.css += ' x-grid-editable-cell';
56840                         }
56841                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56842                             p.css +=  ' x-grid-dirty-cell';
56843                         }
56844                         var markup = ct.apply(p);
56845                         if(!c.locked){
56846                             cb+= markup;
56847                         }else{
56848                             lcb+= markup;
56849                         }
56850                     }
56851                     var alt = [];
56852                     if(stripe && ((rowIndex+1) % 2 == 0)){
56853                         alt.push("x-grid-row-alt")
56854                     }
56855                     if(r.dirty){
56856                         alt.push(  " x-grid-dirty-row");
56857                     }
56858                     rp.cells = lcb;
56859                     if(this.getRowClass){
56860                         alt.push(this.getRowClass(r, rowIndex));
56861                     }
56862                     if (hasListener) {
56863                         rowcfg = {
56864                              
56865                             record: r,
56866                             rowIndex : rowIndex,
56867                             rowClass : ''
56868                         };
56869                         this.grid.fireEvent('rowclass', this, rowcfg);
56870                         alt.push(rowcfg.rowClass);
56871                     }
56872                     rp.alt = alt.join(" ");
56873                     lbuf+= rt.apply(rp);
56874                     rp.cells = cb;
56875                     buf+=  rt.apply(rp);
56876                 }
56877                 return [lbuf, buf];
56878             } :
56879             function(cs, rs, ds, startRow, colCount, stripe){
56880                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56881                 // buffers
56882                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56883                 var hasListener = this.grid.hasListener('rowclass');
56884  
56885                 var rowcfg = {};
56886                 for(var j = 0, len = rs.length; j < len; j++){
56887                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56888                     for(var i = 0; i < colCount; i++){
56889                         c = cs[i];
56890                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56891                         p.id = c.id;
56892                         p.css = p.attr = "";
56893                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56894                         if(p.value == undefined || p.value === "") {
56895                             p.value = "&#160;";
56896                         }
56897                         //Roo.log(c);
56898                          if(c.has_editor){
56899                             p.css += ' x-grid-editable-cell';
56900                         }
56901                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56902                             p.css += ' x-grid-dirty-cell' 
56903                         }
56904                         
56905                         var markup = ct.apply(p);
56906                         if(!c.locked){
56907                             cb[cb.length] = markup;
56908                         }else{
56909                             lcb[lcb.length] = markup;
56910                         }
56911                     }
56912                     var alt = [];
56913                     if(stripe && ((rowIndex+1) % 2 == 0)){
56914                         alt.push( "x-grid-row-alt");
56915                     }
56916                     if(r.dirty){
56917                         alt.push(" x-grid-dirty-row");
56918                     }
56919                     rp.cells = lcb;
56920                     if(this.getRowClass){
56921                         alt.push( this.getRowClass(r, rowIndex));
56922                     }
56923                     if (hasListener) {
56924                         rowcfg = {
56925                              
56926                             record: r,
56927                             rowIndex : rowIndex,
56928                             rowClass : ''
56929                         };
56930                         this.grid.fireEvent('rowclass', this, rowcfg);
56931                         alt.push(rowcfg.rowClass);
56932                     }
56933                     
56934                     rp.alt = alt.join(" ");
56935                     rp.cells = lcb.join("");
56936                     lbuf[lbuf.length] = rt.apply(rp);
56937                     rp.cells = cb.join("");
56938                     buf[buf.length] =  rt.apply(rp);
56939                 }
56940                 return [lbuf.join(""), buf.join("")];
56941             },
56942
56943     renderBody : function(){
56944         var markup = this.renderRows();
56945         var bt = this.templates.body;
56946         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56947     },
56948
56949     /**
56950      * Refreshes the grid
56951      * @param {Boolean} headersToo
56952      */
56953     refresh : function(headersToo){
56954         this.fireEvent("beforerefresh", this);
56955         this.grid.stopEditing();
56956         var result = this.renderBody();
56957         this.lockedBody.update(result[0]);
56958         this.mainBody.update(result[1]);
56959         if(headersToo === true){
56960             this.updateHeaders();
56961             this.updateColumns();
56962             this.updateSplitters();
56963             this.updateHeaderSortState();
56964         }
56965         this.syncRowHeights();
56966         this.layout();
56967         this.fireEvent("refresh", this);
56968     },
56969
56970     handleColumnMove : function(cm, oldIndex, newIndex){
56971         this.indexMap = null;
56972         var s = this.getScrollState();
56973         this.refresh(true);
56974         this.restoreScroll(s);
56975         this.afterMove(newIndex);
56976     },
56977
56978     afterMove : function(colIndex){
56979         if(this.enableMoveAnim && Roo.enableFx){
56980             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56981         }
56982         // if multisort - fix sortOrder, and reload..
56983         if (this.grid.dataSource.multiSort) {
56984             // the we can call sort again..
56985             var dm = this.grid.dataSource;
56986             var cm = this.grid.colModel;
56987             var so = [];
56988             for(var i = 0; i < cm.config.length; i++ ) {
56989                 
56990                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56991                     continue; // dont' bother, it's not in sort list or being set.
56992                 }
56993                 
56994                 so.push(cm.config[i].dataIndex);
56995             };
56996             dm.sortOrder = so;
56997             dm.load(dm.lastOptions);
56998             
56999             
57000         }
57001         
57002     },
57003
57004     updateCell : function(dm, rowIndex, dataIndex){
57005         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
57006         if(typeof colIndex == "undefined"){ // not present in grid
57007             return;
57008         }
57009         var cm = this.grid.colModel;
57010         var cell = this.getCell(rowIndex, colIndex);
57011         var cellText = this.getCellText(rowIndex, colIndex);
57012
57013         var p = {
57014             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
57015             id : cm.getColumnId(colIndex),
57016             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
57017         };
57018         var renderer = cm.getRenderer(colIndex);
57019         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
57020         if(typeof val == "undefined" || val === "") {
57021             val = "&#160;";
57022         }
57023         cellText.innerHTML = val;
57024         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
57025         this.syncRowHeights(rowIndex, rowIndex);
57026     },
57027
57028     calcColumnWidth : function(colIndex, maxRowsToMeasure){
57029         var maxWidth = 0;
57030         if(this.grid.autoSizeHeaders){
57031             var h = this.getHeaderCellMeasure(colIndex);
57032             maxWidth = Math.max(maxWidth, h.scrollWidth);
57033         }
57034         var tb, index;
57035         if(this.cm.isLocked(colIndex)){
57036             tb = this.getLockedTable();
57037             index = colIndex;
57038         }else{
57039             tb = this.getBodyTable();
57040             index = colIndex - this.cm.getLockedCount();
57041         }
57042         if(tb && tb.rows){
57043             var rows = tb.rows;
57044             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
57045             for(var i = 0; i < stopIndex; i++){
57046                 var cell = rows[i].childNodes[index].firstChild;
57047                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
57048             }
57049         }
57050         return maxWidth + /*margin for error in IE*/ 5;
57051     },
57052     /**
57053      * Autofit a column to its content.
57054      * @param {Number} colIndex
57055      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
57056      */
57057      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
57058          if(this.cm.isHidden(colIndex)){
57059              return; // can't calc a hidden column
57060          }
57061         if(forceMinSize){
57062             var cid = this.cm.getColumnId(colIndex);
57063             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57064            if(this.grid.autoSizeHeaders){
57065                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57066            }
57067         }
57068         var newWidth = this.calcColumnWidth(colIndex);
57069         this.cm.setColumnWidth(colIndex,
57070             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57071         if(!suppressEvent){
57072             this.grid.fireEvent("columnresize", colIndex, newWidth);
57073         }
57074     },
57075
57076     /**
57077      * Autofits all columns to their content and then expands to fit any extra space in the grid
57078      */
57079      autoSizeColumns : function(){
57080         var cm = this.grid.colModel;
57081         var colCount = cm.getColumnCount();
57082         for(var i = 0; i < colCount; i++){
57083             this.autoSizeColumn(i, true, true);
57084         }
57085         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57086             this.fitColumns();
57087         }else{
57088             this.updateColumns();
57089             this.layout();
57090         }
57091     },
57092
57093     /**
57094      * Autofits all columns to the grid's width proportionate with their current size
57095      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57096      */
57097     fitColumns : function(reserveScrollSpace){
57098         var cm = this.grid.colModel;
57099         var colCount = cm.getColumnCount();
57100         var cols = [];
57101         var width = 0;
57102         var i, w;
57103         for (i = 0; i < colCount; i++){
57104             if(!cm.isHidden(i) && !cm.isFixed(i)){
57105                 w = cm.getColumnWidth(i);
57106                 cols.push(i);
57107                 cols.push(w);
57108                 width += w;
57109             }
57110         }
57111         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57112         if(reserveScrollSpace){
57113             avail -= 17;
57114         }
57115         var frac = (avail - cm.getTotalWidth())/width;
57116         while (cols.length){
57117             w = cols.pop();
57118             i = cols.pop();
57119             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57120         }
57121         this.updateColumns();
57122         this.layout();
57123     },
57124
57125     onRowSelect : function(rowIndex){
57126         var row = this.getRowComposite(rowIndex);
57127         row.addClass("x-grid-row-selected");
57128     },
57129
57130     onRowDeselect : function(rowIndex){
57131         var row = this.getRowComposite(rowIndex);
57132         row.removeClass("x-grid-row-selected");
57133     },
57134
57135     onCellSelect : function(row, col){
57136         var cell = this.getCell(row, col);
57137         if(cell){
57138             Roo.fly(cell).addClass("x-grid-cell-selected");
57139         }
57140     },
57141
57142     onCellDeselect : function(row, col){
57143         var cell = this.getCell(row, col);
57144         if(cell){
57145             Roo.fly(cell).removeClass("x-grid-cell-selected");
57146         }
57147     },
57148
57149     updateHeaderSortState : function(){
57150         
57151         // sort state can be single { field: xxx, direction : yyy}
57152         // or   { xxx=>ASC , yyy : DESC ..... }
57153         
57154         var mstate = {};
57155         if (!this.ds.multiSort) { 
57156             var state = this.ds.getSortState();
57157             if(!state){
57158                 return;
57159             }
57160             mstate[state.field] = state.direction;
57161             // FIXME... - this is not used here.. but might be elsewhere..
57162             this.sortState = state;
57163             
57164         } else {
57165             mstate = this.ds.sortToggle;
57166         }
57167         //remove existing sort classes..
57168         
57169         var sc = this.sortClasses;
57170         var hds = this.el.select(this.headerSelector).removeClass(sc);
57171         
57172         for(var f in mstate) {
57173         
57174             var sortColumn = this.cm.findColumnIndex(f);
57175             
57176             if(sortColumn != -1){
57177                 var sortDir = mstate[f];        
57178                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57179             }
57180         }
57181         
57182          
57183         
57184     },
57185
57186
57187     handleHeaderClick : function(g, index,e){
57188         
57189         Roo.log("header click");
57190         
57191         if (Roo.isTouch) {
57192             // touch events on header are handled by context
57193             this.handleHdCtx(g,index,e);
57194             return;
57195         }
57196         
57197         
57198         if(this.headersDisabled){
57199             return;
57200         }
57201         var dm = g.dataSource, cm = g.colModel;
57202         if(!cm.isSortable(index)){
57203             return;
57204         }
57205         g.stopEditing();
57206         
57207         if (dm.multiSort) {
57208             // update the sortOrder
57209             var so = [];
57210             for(var i = 0; i < cm.config.length; i++ ) {
57211                 
57212                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57213                     continue; // dont' bother, it's not in sort list or being set.
57214                 }
57215                 
57216                 so.push(cm.config[i].dataIndex);
57217             };
57218             dm.sortOrder = so;
57219         }
57220         
57221         
57222         dm.sort(cm.getDataIndex(index));
57223     },
57224
57225
57226     destroy : function(){
57227         if(this.colMenu){
57228             this.colMenu.removeAll();
57229             Roo.menu.MenuMgr.unregister(this.colMenu);
57230             this.colMenu.getEl().remove();
57231             delete this.colMenu;
57232         }
57233         if(this.hmenu){
57234             this.hmenu.removeAll();
57235             Roo.menu.MenuMgr.unregister(this.hmenu);
57236             this.hmenu.getEl().remove();
57237             delete this.hmenu;
57238         }
57239         if(this.grid.enableColumnMove){
57240             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57241             if(dds){
57242                 for(var dd in dds){
57243                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57244                         var elid = dds[dd].dragElId;
57245                         dds[dd].unreg();
57246                         Roo.get(elid).remove();
57247                     } else if(dds[dd].config.isTarget){
57248                         dds[dd].proxyTop.remove();
57249                         dds[dd].proxyBottom.remove();
57250                         dds[dd].unreg();
57251                     }
57252                     if(Roo.dd.DDM.locationCache[dd]){
57253                         delete Roo.dd.DDM.locationCache[dd];
57254                     }
57255                 }
57256                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57257             }
57258         }
57259         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57260         this.bind(null, null);
57261         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57262     },
57263
57264     handleLockChange : function(){
57265         this.refresh(true);
57266     },
57267
57268     onDenyColumnLock : function(){
57269
57270     },
57271
57272     onDenyColumnHide : function(){
57273
57274     },
57275
57276     handleHdMenuClick : function(item){
57277         var index = this.hdCtxIndex;
57278         var cm = this.cm, ds = this.ds;
57279         switch(item.id){
57280             case "asc":
57281                 ds.sort(cm.getDataIndex(index), "ASC");
57282                 break;
57283             case "desc":
57284                 ds.sort(cm.getDataIndex(index), "DESC");
57285                 break;
57286             case "lock":
57287                 var lc = cm.getLockedCount();
57288                 if(cm.getColumnCount(true) <= lc+1){
57289                     this.onDenyColumnLock();
57290                     return;
57291                 }
57292                 if(lc != index){
57293                     cm.setLocked(index, true, true);
57294                     cm.moveColumn(index, lc);
57295                     this.grid.fireEvent("columnmove", index, lc);
57296                 }else{
57297                     cm.setLocked(index, true);
57298                 }
57299             break;
57300             case "unlock":
57301                 var lc = cm.getLockedCount();
57302                 if((lc-1) != index){
57303                     cm.setLocked(index, false, true);
57304                     cm.moveColumn(index, lc-1);
57305                     this.grid.fireEvent("columnmove", index, lc-1);
57306                 }else{
57307                     cm.setLocked(index, false);
57308                 }
57309             break;
57310             case 'wider': // used to expand cols on touch..
57311             case 'narrow':
57312                 var cw = cm.getColumnWidth(index);
57313                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57314                 cw = Math.max(0, cw);
57315                 cw = Math.min(cw,4000);
57316                 cm.setColumnWidth(index, cw);
57317                 break;
57318                 
57319             default:
57320                 index = cm.getIndexById(item.id.substr(4));
57321                 if(index != -1){
57322                     if(item.checked && cm.getColumnCount(true) <= 1){
57323                         this.onDenyColumnHide();
57324                         return false;
57325                     }
57326                     cm.setHidden(index, item.checked);
57327                 }
57328         }
57329         return true;
57330     },
57331
57332     beforeColMenuShow : function(){
57333         var cm = this.cm,  colCount = cm.getColumnCount();
57334         this.colMenu.removeAll();
57335         for(var i = 0; i < colCount; i++){
57336             this.colMenu.add(new Roo.menu.CheckItem({
57337                 id: "col-"+cm.getColumnId(i),
57338                 text: cm.getColumnHeader(i),
57339                 checked: !cm.isHidden(i),
57340                 hideOnClick:false
57341             }));
57342         }
57343     },
57344
57345     handleHdCtx : function(g, index, e){
57346         e.stopEvent();
57347         var hd = this.getHeaderCell(index);
57348         this.hdCtxIndex = index;
57349         var ms = this.hmenu.items, cm = this.cm;
57350         ms.get("asc").setDisabled(!cm.isSortable(index));
57351         ms.get("desc").setDisabled(!cm.isSortable(index));
57352         if(this.grid.enableColLock !== false){
57353             ms.get("lock").setDisabled(cm.isLocked(index));
57354             ms.get("unlock").setDisabled(!cm.isLocked(index));
57355         }
57356         this.hmenu.show(hd, "tl-bl");
57357     },
57358
57359     handleHdOver : function(e){
57360         var hd = this.findHeaderCell(e.getTarget());
57361         if(hd && !this.headersDisabled){
57362             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57363                this.fly(hd).addClass("x-grid-hd-over");
57364             }
57365         }
57366     },
57367
57368     handleHdOut : function(e){
57369         var hd = this.findHeaderCell(e.getTarget());
57370         if(hd){
57371             this.fly(hd).removeClass("x-grid-hd-over");
57372         }
57373     },
57374
57375     handleSplitDblClick : function(e, t){
57376         var i = this.getCellIndex(t);
57377         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57378             this.autoSizeColumn(i, true);
57379             this.layout();
57380         }
57381     },
57382
57383     render : function(){
57384
57385         var cm = this.cm;
57386         var colCount = cm.getColumnCount();
57387
57388         if(this.grid.monitorWindowResize === true){
57389             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57390         }
57391         var header = this.renderHeaders();
57392         var body = this.templates.body.apply({rows:""});
57393         var html = this.templates.master.apply({
57394             lockedBody: body,
57395             body: body,
57396             lockedHeader: header[0],
57397             header: header[1]
57398         });
57399
57400         //this.updateColumns();
57401
57402         this.grid.getGridEl().dom.innerHTML = html;
57403
57404         this.initElements();
57405         
57406         // a kludge to fix the random scolling effect in webkit
57407         this.el.on("scroll", function() {
57408             this.el.dom.scrollTop=0; // hopefully not recursive..
57409         },this);
57410
57411         this.scroller.on("scroll", this.handleScroll, this);
57412         this.lockedBody.on("mousewheel", this.handleWheel, this);
57413         this.mainBody.on("mousewheel", this.handleWheel, this);
57414
57415         this.mainHd.on("mouseover", this.handleHdOver, this);
57416         this.mainHd.on("mouseout", this.handleHdOut, this);
57417         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57418                 {delegate: "."+this.splitClass});
57419
57420         this.lockedHd.on("mouseover", this.handleHdOver, this);
57421         this.lockedHd.on("mouseout", this.handleHdOut, this);
57422         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57423                 {delegate: "."+this.splitClass});
57424
57425         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57426             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57427         }
57428
57429         this.updateSplitters();
57430
57431         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57432             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57433             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57434         }
57435
57436         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57437             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57438             this.hmenu.add(
57439                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57440                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57441             );
57442             if(this.grid.enableColLock !== false){
57443                 this.hmenu.add('-',
57444                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57445                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57446                 );
57447             }
57448             if (Roo.isTouch) {
57449                  this.hmenu.add('-',
57450                     {id:"wider", text: this.columnsWiderText},
57451                     {id:"narrow", text: this.columnsNarrowText }
57452                 );
57453                 
57454                  
57455             }
57456             
57457             if(this.grid.enableColumnHide !== false){
57458
57459                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57460                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57461                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57462
57463                 this.hmenu.add('-',
57464                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57465                 );
57466             }
57467             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57468
57469             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57470         }
57471
57472         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57473             this.dd = new Roo.grid.GridDragZone(this.grid, {
57474                 ddGroup : this.grid.ddGroup || 'GridDD'
57475             });
57476             
57477         }
57478
57479         /*
57480         for(var i = 0; i < colCount; i++){
57481             if(cm.isHidden(i)){
57482                 this.hideColumn(i);
57483             }
57484             if(cm.config[i].align){
57485                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57486                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57487             }
57488         }*/
57489         
57490         this.updateHeaderSortState();
57491
57492         this.beforeInitialResize();
57493         this.layout(true);
57494
57495         // two part rendering gives faster view to the user
57496         this.renderPhase2.defer(1, this);
57497     },
57498
57499     renderPhase2 : function(){
57500         // render the rows now
57501         this.refresh();
57502         if(this.grid.autoSizeColumns){
57503             this.autoSizeColumns();
57504         }
57505     },
57506
57507     beforeInitialResize : function(){
57508
57509     },
57510
57511     onColumnSplitterMoved : function(i, w){
57512         this.userResized = true;
57513         var cm = this.grid.colModel;
57514         cm.setColumnWidth(i, w, true);
57515         var cid = cm.getColumnId(i);
57516         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57517         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57518         this.updateSplitters();
57519         this.layout();
57520         this.grid.fireEvent("columnresize", i, w);
57521     },
57522
57523     syncRowHeights : function(startIndex, endIndex){
57524         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57525             startIndex = startIndex || 0;
57526             var mrows = this.getBodyTable().rows;
57527             var lrows = this.getLockedTable().rows;
57528             var len = mrows.length-1;
57529             endIndex = Math.min(endIndex || len, len);
57530             for(var i = startIndex; i <= endIndex; i++){
57531                 var m = mrows[i], l = lrows[i];
57532                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57533                 m.style.height = l.style.height = h + "px";
57534             }
57535         }
57536     },
57537
57538     layout : function(initialRender, is2ndPass){
57539         var g = this.grid;
57540         var auto = g.autoHeight;
57541         var scrollOffset = 16;
57542         var c = g.getGridEl(), cm = this.cm,
57543                 expandCol = g.autoExpandColumn,
57544                 gv = this;
57545         //c.beginMeasure();
57546
57547         if(!c.dom.offsetWidth){ // display:none?
57548             if(initialRender){
57549                 this.lockedWrap.show();
57550                 this.mainWrap.show();
57551             }
57552             return;
57553         }
57554
57555         var hasLock = this.cm.isLocked(0);
57556
57557         var tbh = this.headerPanel.getHeight();
57558         var bbh = this.footerPanel.getHeight();
57559
57560         if(auto){
57561             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57562             var newHeight = ch + c.getBorderWidth("tb");
57563             if(g.maxHeight){
57564                 newHeight = Math.min(g.maxHeight, newHeight);
57565             }
57566             c.setHeight(newHeight);
57567         }
57568
57569         if(g.autoWidth){
57570             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57571         }
57572
57573         var s = this.scroller;
57574
57575         var csize = c.getSize(true);
57576
57577         this.el.setSize(csize.width, csize.height);
57578
57579         this.headerPanel.setWidth(csize.width);
57580         this.footerPanel.setWidth(csize.width);
57581
57582         var hdHeight = this.mainHd.getHeight();
57583         var vw = csize.width;
57584         var vh = csize.height - (tbh + bbh);
57585
57586         s.setSize(vw, vh);
57587
57588         var bt = this.getBodyTable();
57589         
57590         if(cm.getLockedCount() == cm.config.length){
57591             bt = this.getLockedTable();
57592         }
57593         
57594         var ltWidth = hasLock ?
57595                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57596
57597         var scrollHeight = bt.offsetHeight;
57598         var scrollWidth = ltWidth + bt.offsetWidth;
57599         var vscroll = false, hscroll = false;
57600
57601         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57602
57603         var lw = this.lockedWrap, mw = this.mainWrap;
57604         var lb = this.lockedBody, mb = this.mainBody;
57605
57606         setTimeout(function(){
57607             var t = s.dom.offsetTop;
57608             var w = s.dom.clientWidth,
57609                 h = s.dom.clientHeight;
57610
57611             lw.setTop(t);
57612             lw.setSize(ltWidth, h);
57613
57614             mw.setLeftTop(ltWidth, t);
57615             mw.setSize(w-ltWidth, h);
57616
57617             lb.setHeight(h-hdHeight);
57618             mb.setHeight(h-hdHeight);
57619
57620             if(is2ndPass !== true && !gv.userResized && expandCol){
57621                 // high speed resize without full column calculation
57622                 
57623                 var ci = cm.getIndexById(expandCol);
57624                 if (ci < 0) {
57625                     ci = cm.findColumnIndex(expandCol);
57626                 }
57627                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57628                 var expandId = cm.getColumnId(ci);
57629                 var  tw = cm.getTotalWidth(false);
57630                 var currentWidth = cm.getColumnWidth(ci);
57631                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57632                 if(currentWidth != cw){
57633                     cm.setColumnWidth(ci, cw, true);
57634                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57635                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57636                     gv.updateSplitters();
57637                     gv.layout(false, true);
57638                 }
57639             }
57640
57641             if(initialRender){
57642                 lw.show();
57643                 mw.show();
57644             }
57645             //c.endMeasure();
57646         }, 10);
57647     },
57648
57649     onWindowResize : function(){
57650         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57651             return;
57652         }
57653         this.layout();
57654     },
57655
57656     appendFooter : function(parentEl){
57657         return null;
57658     },
57659
57660     sortAscText : "Sort Ascending",
57661     sortDescText : "Sort Descending",
57662     lockText : "Lock Column",
57663     unlockText : "Unlock Column",
57664     columnsText : "Columns",
57665  
57666     columnsWiderText : "Wider",
57667     columnsNarrowText : "Thinner"
57668 });
57669
57670
57671 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57672     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57673     this.proxy.el.addClass('x-grid3-col-dd');
57674 };
57675
57676 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57677     handleMouseDown : function(e){
57678
57679     },
57680
57681     callHandleMouseDown : function(e){
57682         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57683     }
57684 });
57685 /*
57686  * Based on:
57687  * Ext JS Library 1.1.1
57688  * Copyright(c) 2006-2007, Ext JS, LLC.
57689  *
57690  * Originally Released Under LGPL - original licence link has changed is not relivant.
57691  *
57692  * Fork - LGPL
57693  * <script type="text/javascript">
57694  */
57695  
57696 // private
57697 // This is a support class used internally by the Grid components
57698 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57699     this.grid = grid;
57700     this.view = grid.getView();
57701     this.proxy = this.view.resizeProxy;
57702     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57703         "gridSplitters" + this.grid.getGridEl().id, {
57704         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57705     });
57706     this.setHandleElId(Roo.id(hd));
57707     this.setOuterHandleElId(Roo.id(hd2));
57708     this.scroll = false;
57709 };
57710 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57711     fly: Roo.Element.fly,
57712
57713     b4StartDrag : function(x, y){
57714         this.view.headersDisabled = true;
57715         this.proxy.setHeight(this.view.mainWrap.getHeight());
57716         var w = this.cm.getColumnWidth(this.cellIndex);
57717         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57718         this.resetConstraints();
57719         this.setXConstraint(minw, 1000);
57720         this.setYConstraint(0, 0);
57721         this.minX = x - minw;
57722         this.maxX = x + 1000;
57723         this.startPos = x;
57724         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57725     },
57726
57727
57728     handleMouseDown : function(e){
57729         ev = Roo.EventObject.setEvent(e);
57730         var t = this.fly(ev.getTarget());
57731         if(t.hasClass("x-grid-split")){
57732             this.cellIndex = this.view.getCellIndex(t.dom);
57733             this.split = t.dom;
57734             this.cm = this.grid.colModel;
57735             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57736                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57737             }
57738         }
57739     },
57740
57741     endDrag : function(e){
57742         this.view.headersDisabled = false;
57743         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57744         var diff = endX - this.startPos;
57745         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57746     },
57747
57748     autoOffset : function(){
57749         this.setDelta(0,0);
57750     }
57751 });/*
57752  * Based on:
57753  * Ext JS Library 1.1.1
57754  * Copyright(c) 2006-2007, Ext JS, LLC.
57755  *
57756  * Originally Released Under LGPL - original licence link has changed is not relivant.
57757  *
57758  * Fork - LGPL
57759  * <script type="text/javascript">
57760  */
57761  
57762 // private
57763 // This is a support class used internally by the Grid components
57764 Roo.grid.GridDragZone = function(grid, config){
57765     this.view = grid.getView();
57766     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57767     if(this.view.lockedBody){
57768         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57769         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57770     }
57771     this.scroll = false;
57772     this.grid = grid;
57773     this.ddel = document.createElement('div');
57774     this.ddel.className = 'x-grid-dd-wrap';
57775 };
57776
57777 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57778     ddGroup : "GridDD",
57779
57780     getDragData : function(e){
57781         var t = Roo.lib.Event.getTarget(e);
57782         var rowIndex = this.view.findRowIndex(t);
57783         var sm = this.grid.selModel;
57784             
57785         //Roo.log(rowIndex);
57786         
57787         if (sm.getSelectedCell) {
57788             // cell selection..
57789             if (!sm.getSelectedCell()) {
57790                 return false;
57791             }
57792             if (rowIndex != sm.getSelectedCell()[0]) {
57793                 return false;
57794             }
57795         
57796         }
57797         
57798         if(rowIndex !== false){
57799             
57800             // if editorgrid.. 
57801             
57802             
57803             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57804                
57805             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57806               //  
57807             //}
57808             if (e.hasModifier()){
57809                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57810             }
57811             
57812             Roo.log("getDragData");
57813             
57814             return {
57815                 grid: this.grid,
57816                 ddel: this.ddel,
57817                 rowIndex: rowIndex,
57818                 selections:sm.getSelections ? sm.getSelections() : (
57819                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57820                 )
57821             };
57822         }
57823         return false;
57824     },
57825
57826     onInitDrag : function(e){
57827         var data = this.dragData;
57828         this.ddel.innerHTML = this.grid.getDragDropText();
57829         this.proxy.update(this.ddel);
57830         // fire start drag?
57831     },
57832
57833     afterRepair : function(){
57834         this.dragging = false;
57835     },
57836
57837     getRepairXY : function(e, data){
57838         return false;
57839     },
57840
57841     onEndDrag : function(data, e){
57842         // fire end drag?
57843     },
57844
57845     onValidDrop : function(dd, e, id){
57846         // fire drag drop?
57847         this.hideProxy();
57848     },
57849
57850     beforeInvalidDrop : function(e, id){
57851
57852     }
57853 });/*
57854  * Based on:
57855  * Ext JS Library 1.1.1
57856  * Copyright(c) 2006-2007, Ext JS, LLC.
57857  *
57858  * Originally Released Under LGPL - original licence link has changed is not relivant.
57859  *
57860  * Fork - LGPL
57861  * <script type="text/javascript">
57862  */
57863  
57864
57865 /**
57866  * @class Roo.grid.ColumnModel
57867  * @extends Roo.util.Observable
57868  * This is the default implementation of a ColumnModel used by the Grid. It defines
57869  * the columns in the grid.
57870  * <br>Usage:<br>
57871  <pre><code>
57872  var colModel = new Roo.grid.ColumnModel([
57873         {header: "Ticker", width: 60, sortable: true, locked: true},
57874         {header: "Company Name", width: 150, sortable: true},
57875         {header: "Market Cap.", width: 100, sortable: true},
57876         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57877         {header: "Employees", width: 100, sortable: true, resizable: false}
57878  ]);
57879  </code></pre>
57880  * <p>
57881  
57882  * The config options listed for this class are options which may appear in each
57883  * individual column definition.
57884  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57885  * @constructor
57886  * @param {Object} config An Array of column config objects. See this class's
57887  * config objects for details.
57888 */
57889 Roo.grid.ColumnModel = function(config){
57890         /**
57891      * The config passed into the constructor
57892      */
57893     this.config = config;
57894     this.lookup = {};
57895
57896     // if no id, create one
57897     // if the column does not have a dataIndex mapping,
57898     // map it to the order it is in the config
57899     for(var i = 0, len = config.length; i < len; i++){
57900         var c = config[i];
57901         if(typeof c.dataIndex == "undefined"){
57902             c.dataIndex = i;
57903         }
57904         if(typeof c.renderer == "string"){
57905             c.renderer = Roo.util.Format[c.renderer];
57906         }
57907         if(typeof c.id == "undefined"){
57908             c.id = Roo.id();
57909         }
57910         if(c.editor && c.editor.xtype){
57911             c.editor  = Roo.factory(c.editor, Roo.grid);
57912         }
57913         if(c.editor && c.editor.isFormField){
57914             c.editor = new Roo.grid.GridEditor(c.editor);
57915         }
57916         this.lookup[c.id] = c;
57917     }
57918
57919     /**
57920      * The width of columns which have no width specified (defaults to 100)
57921      * @type Number
57922      */
57923     this.defaultWidth = 100;
57924
57925     /**
57926      * Default sortable of columns which have no sortable specified (defaults to false)
57927      * @type Boolean
57928      */
57929     this.defaultSortable = false;
57930
57931     this.addEvents({
57932         /**
57933              * @event widthchange
57934              * Fires when the width of a column changes.
57935              * @param {ColumnModel} this
57936              * @param {Number} columnIndex The column index
57937              * @param {Number} newWidth The new width
57938              */
57939             "widthchange": true,
57940         /**
57941              * @event headerchange
57942              * Fires when the text of a header changes.
57943              * @param {ColumnModel} this
57944              * @param {Number} columnIndex The column index
57945              * @param {Number} newText The new header text
57946              */
57947             "headerchange": true,
57948         /**
57949              * @event hiddenchange
57950              * Fires when a column is hidden or "unhidden".
57951              * @param {ColumnModel} this
57952              * @param {Number} columnIndex The column index
57953              * @param {Boolean} hidden true if hidden, false otherwise
57954              */
57955             "hiddenchange": true,
57956             /**
57957          * @event columnmoved
57958          * Fires when a column is moved.
57959          * @param {ColumnModel} this
57960          * @param {Number} oldIndex
57961          * @param {Number} newIndex
57962          */
57963         "columnmoved" : true,
57964         /**
57965          * @event columlockchange
57966          * Fires when a column's locked state is changed
57967          * @param {ColumnModel} this
57968          * @param {Number} colIndex
57969          * @param {Boolean} locked true if locked
57970          */
57971         "columnlockchange" : true
57972     });
57973     Roo.grid.ColumnModel.superclass.constructor.call(this);
57974 };
57975 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57976     /**
57977      * @cfg {String} header The header text to display in the Grid view.
57978      */
57979     /**
57980      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57981      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57982      * specified, the column's index is used as an index into the Record's data Array.
57983      */
57984     /**
57985      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57986      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57987      */
57988     /**
57989      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57990      * Defaults to the value of the {@link #defaultSortable} property.
57991      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57992      */
57993     /**
57994      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57995      */
57996     /**
57997      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57998      */
57999     /**
58000      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
58001      */
58002     /**
58003      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
58004      */
58005     /**
58006      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
58007      * given the cell's data value. See {@link #setRenderer}. If not specified, the
58008      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
58009      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
58010      */
58011        /**
58012      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
58013      */
58014     /**
58015      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
58016      */
58017     /**
58018      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
58019      */
58020     /**
58021      * @cfg {String} cursor (Optional)
58022      */
58023     /**
58024      * @cfg {String} tooltip (Optional)
58025      */
58026     /**
58027      * @cfg {Number} xs (Optional)
58028      */
58029     /**
58030      * @cfg {Number} sm (Optional)
58031      */
58032     /**
58033      * @cfg {Number} md (Optional)
58034      */
58035     /**
58036      * @cfg {Number} lg (Optional)
58037      */
58038     /**
58039      * Returns the id of the column at the specified index.
58040      * @param {Number} index The column index
58041      * @return {String} the id
58042      */
58043     getColumnId : function(index){
58044         return this.config[index].id;
58045     },
58046
58047     /**
58048      * Returns the column for a specified id.
58049      * @param {String} id The column id
58050      * @return {Object} the column
58051      */
58052     getColumnById : function(id){
58053         return this.lookup[id];
58054     },
58055
58056     
58057     /**
58058      * Returns the column for a specified dataIndex.
58059      * @param {String} dataIndex The column dataIndex
58060      * @return {Object|Boolean} the column or false if not found
58061      */
58062     getColumnByDataIndex: function(dataIndex){
58063         var index = this.findColumnIndex(dataIndex);
58064         return index > -1 ? this.config[index] : false;
58065     },
58066     
58067     /**
58068      * Returns the index for a specified column id.
58069      * @param {String} id The column id
58070      * @return {Number} the index, or -1 if not found
58071      */
58072     getIndexById : function(id){
58073         for(var i = 0, len = this.config.length; i < len; i++){
58074             if(this.config[i].id == id){
58075                 return i;
58076             }
58077         }
58078         return -1;
58079     },
58080     
58081     /**
58082      * Returns the index for a specified column dataIndex.
58083      * @param {String} dataIndex The column dataIndex
58084      * @return {Number} the index, or -1 if not found
58085      */
58086     
58087     findColumnIndex : function(dataIndex){
58088         for(var i = 0, len = this.config.length; i < len; i++){
58089             if(this.config[i].dataIndex == dataIndex){
58090                 return i;
58091             }
58092         }
58093         return -1;
58094     },
58095     
58096     
58097     moveColumn : function(oldIndex, newIndex){
58098         var c = this.config[oldIndex];
58099         this.config.splice(oldIndex, 1);
58100         this.config.splice(newIndex, 0, c);
58101         this.dataMap = null;
58102         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58103     },
58104
58105     isLocked : function(colIndex){
58106         return this.config[colIndex].locked === true;
58107     },
58108
58109     setLocked : function(colIndex, value, suppressEvent){
58110         if(this.isLocked(colIndex) == value){
58111             return;
58112         }
58113         this.config[colIndex].locked = value;
58114         if(!suppressEvent){
58115             this.fireEvent("columnlockchange", this, colIndex, value);
58116         }
58117     },
58118
58119     getTotalLockedWidth : function(){
58120         var totalWidth = 0;
58121         for(var i = 0; i < this.config.length; i++){
58122             if(this.isLocked(i) && !this.isHidden(i)){
58123                 this.totalWidth += this.getColumnWidth(i);
58124             }
58125         }
58126         return totalWidth;
58127     },
58128
58129     getLockedCount : function(){
58130         for(var i = 0, len = this.config.length; i < len; i++){
58131             if(!this.isLocked(i)){
58132                 return i;
58133             }
58134         }
58135         
58136         return this.config.length;
58137     },
58138
58139     /**
58140      * Returns the number of columns.
58141      * @return {Number}
58142      */
58143     getColumnCount : function(visibleOnly){
58144         if(visibleOnly === true){
58145             var c = 0;
58146             for(var i = 0, len = this.config.length; i < len; i++){
58147                 if(!this.isHidden(i)){
58148                     c++;
58149                 }
58150             }
58151             return c;
58152         }
58153         return this.config.length;
58154     },
58155
58156     /**
58157      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58158      * @param {Function} fn
58159      * @param {Object} scope (optional)
58160      * @return {Array} result
58161      */
58162     getColumnsBy : function(fn, scope){
58163         var r = [];
58164         for(var i = 0, len = this.config.length; i < len; i++){
58165             var c = this.config[i];
58166             if(fn.call(scope||this, c, i) === true){
58167                 r[r.length] = c;
58168             }
58169         }
58170         return r;
58171     },
58172
58173     /**
58174      * Returns true if the specified column is sortable.
58175      * @param {Number} col The column index
58176      * @return {Boolean}
58177      */
58178     isSortable : function(col){
58179         if(typeof this.config[col].sortable == "undefined"){
58180             return this.defaultSortable;
58181         }
58182         return this.config[col].sortable;
58183     },
58184
58185     /**
58186      * Returns the rendering (formatting) function defined for the column.
58187      * @param {Number} col The column index.
58188      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58189      */
58190     getRenderer : function(col){
58191         if(!this.config[col].renderer){
58192             return Roo.grid.ColumnModel.defaultRenderer;
58193         }
58194         return this.config[col].renderer;
58195     },
58196
58197     /**
58198      * Sets the rendering (formatting) function for a column.
58199      * @param {Number} col The column index
58200      * @param {Function} fn The function to use to process the cell's raw data
58201      * to return HTML markup for the grid view. The render function is called with
58202      * the following parameters:<ul>
58203      * <li>Data value.</li>
58204      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58205      * <li>css A CSS style string to apply to the table cell.</li>
58206      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58207      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58208      * <li>Row index</li>
58209      * <li>Column index</li>
58210      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58211      */
58212     setRenderer : function(col, fn){
58213         this.config[col].renderer = fn;
58214     },
58215
58216     /**
58217      * Returns the width for the specified column.
58218      * @param {Number} col The column index
58219      * @return {Number}
58220      */
58221     getColumnWidth : function(col){
58222         return this.config[col].width * 1 || this.defaultWidth;
58223     },
58224
58225     /**
58226      * Sets the width for a column.
58227      * @param {Number} col The column index
58228      * @param {Number} width The new width
58229      */
58230     setColumnWidth : function(col, width, suppressEvent){
58231         this.config[col].width = width;
58232         this.totalWidth = null;
58233         if(!suppressEvent){
58234              this.fireEvent("widthchange", this, col, width);
58235         }
58236     },
58237
58238     /**
58239      * Returns the total width of all columns.
58240      * @param {Boolean} includeHidden True to include hidden column widths
58241      * @return {Number}
58242      */
58243     getTotalWidth : function(includeHidden){
58244         if(!this.totalWidth){
58245             this.totalWidth = 0;
58246             for(var i = 0, len = this.config.length; i < len; i++){
58247                 if(includeHidden || !this.isHidden(i)){
58248                     this.totalWidth += this.getColumnWidth(i);
58249                 }
58250             }
58251         }
58252         return this.totalWidth;
58253     },
58254
58255     /**
58256      * Returns the header for the specified column.
58257      * @param {Number} col The column index
58258      * @return {String}
58259      */
58260     getColumnHeader : function(col){
58261         return this.config[col].header;
58262     },
58263
58264     /**
58265      * Sets the header for a column.
58266      * @param {Number} col The column index
58267      * @param {String} header The new header
58268      */
58269     setColumnHeader : function(col, header){
58270         this.config[col].header = header;
58271         this.fireEvent("headerchange", this, col, header);
58272     },
58273
58274     /**
58275      * Returns the tooltip for the specified column.
58276      * @param {Number} col The column index
58277      * @return {String}
58278      */
58279     getColumnTooltip : function(col){
58280             return this.config[col].tooltip;
58281     },
58282     /**
58283      * Sets the tooltip for a column.
58284      * @param {Number} col The column index
58285      * @param {String} tooltip The new tooltip
58286      */
58287     setColumnTooltip : function(col, tooltip){
58288             this.config[col].tooltip = tooltip;
58289     },
58290
58291     /**
58292      * Returns the dataIndex for the specified column.
58293      * @param {Number} col The column index
58294      * @return {Number}
58295      */
58296     getDataIndex : function(col){
58297         return this.config[col].dataIndex;
58298     },
58299
58300     /**
58301      * Sets the dataIndex for a column.
58302      * @param {Number} col The column index
58303      * @param {Number} dataIndex The new dataIndex
58304      */
58305     setDataIndex : function(col, dataIndex){
58306         this.config[col].dataIndex = dataIndex;
58307     },
58308
58309     
58310     
58311     /**
58312      * Returns true if the cell is editable.
58313      * @param {Number} colIndex The column index
58314      * @param {Number} rowIndex The row index - this is nto actually used..?
58315      * @return {Boolean}
58316      */
58317     isCellEditable : function(colIndex, rowIndex){
58318         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58319     },
58320
58321     /**
58322      * Returns the editor defined for the cell/column.
58323      * return false or null to disable editing.
58324      * @param {Number} colIndex The column index
58325      * @param {Number} rowIndex The row index
58326      * @return {Object}
58327      */
58328     getCellEditor : function(colIndex, rowIndex){
58329         return this.config[colIndex].editor;
58330     },
58331
58332     /**
58333      * Sets if a column is editable.
58334      * @param {Number} col The column index
58335      * @param {Boolean} editable True if the column is editable
58336      */
58337     setEditable : function(col, editable){
58338         this.config[col].editable = editable;
58339     },
58340
58341
58342     /**
58343      * Returns true if the column is hidden.
58344      * @param {Number} colIndex The column index
58345      * @return {Boolean}
58346      */
58347     isHidden : function(colIndex){
58348         return this.config[colIndex].hidden;
58349     },
58350
58351
58352     /**
58353      * Returns true if the column width cannot be changed
58354      */
58355     isFixed : function(colIndex){
58356         return this.config[colIndex].fixed;
58357     },
58358
58359     /**
58360      * Returns true if the column can be resized
58361      * @return {Boolean}
58362      */
58363     isResizable : function(colIndex){
58364         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58365     },
58366     /**
58367      * Sets if a column is hidden.
58368      * @param {Number} colIndex The column index
58369      * @param {Boolean} hidden True if the column is hidden
58370      */
58371     setHidden : function(colIndex, hidden){
58372         this.config[colIndex].hidden = hidden;
58373         this.totalWidth = null;
58374         this.fireEvent("hiddenchange", this, colIndex, hidden);
58375     },
58376
58377     /**
58378      * Sets the editor for a column.
58379      * @param {Number} col The column index
58380      * @param {Object} editor The editor object
58381      */
58382     setEditor : function(col, editor){
58383         this.config[col].editor = editor;
58384     }
58385 });
58386
58387 Roo.grid.ColumnModel.defaultRenderer = function(value)
58388 {
58389     if(typeof value == "object") {
58390         return value;
58391     }
58392         if(typeof value == "string" && value.length < 1){
58393             return "&#160;";
58394         }
58395     
58396         return String.format("{0}", value);
58397 };
58398
58399 // Alias for backwards compatibility
58400 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58401 /*
58402  * Based on:
58403  * Ext JS Library 1.1.1
58404  * Copyright(c) 2006-2007, Ext JS, LLC.
58405  *
58406  * Originally Released Under LGPL - original licence link has changed is not relivant.
58407  *
58408  * Fork - LGPL
58409  * <script type="text/javascript">
58410  */
58411
58412 /**
58413  * @class Roo.grid.AbstractSelectionModel
58414  * @extends Roo.util.Observable
58415  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58416  * implemented by descendant classes.  This class should not be directly instantiated.
58417  * @constructor
58418  */
58419 Roo.grid.AbstractSelectionModel = function(){
58420     this.locked = false;
58421     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58422 };
58423
58424 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58425     /** @ignore Called by the grid automatically. Do not call directly. */
58426     init : function(grid){
58427         this.grid = grid;
58428         this.initEvents();
58429     },
58430
58431     /**
58432      * Locks the selections.
58433      */
58434     lock : function(){
58435         this.locked = true;
58436     },
58437
58438     /**
58439      * Unlocks the selections.
58440      */
58441     unlock : function(){
58442         this.locked = false;
58443     },
58444
58445     /**
58446      * Returns true if the selections are locked.
58447      * @return {Boolean}
58448      */
58449     isLocked : function(){
58450         return this.locked;
58451     }
58452 });/*
58453  * Based on:
58454  * Ext JS Library 1.1.1
58455  * Copyright(c) 2006-2007, Ext JS, LLC.
58456  *
58457  * Originally Released Under LGPL - original licence link has changed is not relivant.
58458  *
58459  * Fork - LGPL
58460  * <script type="text/javascript">
58461  */
58462 /**
58463  * @extends Roo.grid.AbstractSelectionModel
58464  * @class Roo.grid.RowSelectionModel
58465  * The default SelectionModel used by {@link Roo.grid.Grid}.
58466  * It supports multiple selections and keyboard selection/navigation. 
58467  * @constructor
58468  * @param {Object} config
58469  */
58470 Roo.grid.RowSelectionModel = function(config){
58471     Roo.apply(this, config);
58472     this.selections = new Roo.util.MixedCollection(false, function(o){
58473         return o.id;
58474     });
58475
58476     this.last = false;
58477     this.lastActive = false;
58478
58479     this.addEvents({
58480         /**
58481              * @event selectionchange
58482              * Fires when the selection changes
58483              * @param {SelectionModel} this
58484              */
58485             "selectionchange" : true,
58486         /**
58487              * @event afterselectionchange
58488              * Fires after the selection changes (eg. by key press or clicking)
58489              * @param {SelectionModel} this
58490              */
58491             "afterselectionchange" : true,
58492         /**
58493              * @event beforerowselect
58494              * Fires when a row is selected being selected, return false to cancel.
58495              * @param {SelectionModel} this
58496              * @param {Number} rowIndex The selected index
58497              * @param {Boolean} keepExisting False if other selections will be cleared
58498              */
58499             "beforerowselect" : true,
58500         /**
58501              * @event rowselect
58502              * Fires when a row is selected.
58503              * @param {SelectionModel} this
58504              * @param {Number} rowIndex The selected index
58505              * @param {Roo.data.Record} r The record
58506              */
58507             "rowselect" : true,
58508         /**
58509              * @event rowdeselect
58510              * Fires when a row is deselected.
58511              * @param {SelectionModel} this
58512              * @param {Number} rowIndex The selected index
58513              */
58514         "rowdeselect" : true
58515     });
58516     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58517     this.locked = false;
58518 };
58519
58520 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58521     /**
58522      * @cfg {Boolean} singleSelect
58523      * True to allow selection of only one row at a time (defaults to false)
58524      */
58525     singleSelect : false,
58526
58527     // private
58528     initEvents : function(){
58529
58530         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58531             this.grid.on("mousedown", this.handleMouseDown, this);
58532         }else{ // allow click to work like normal
58533             this.grid.on("rowclick", this.handleDragableRowClick, this);
58534         }
58535
58536         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58537             "up" : function(e){
58538                 if(!e.shiftKey){
58539                     this.selectPrevious(e.shiftKey);
58540                 }else if(this.last !== false && this.lastActive !== false){
58541                     var last = this.last;
58542                     this.selectRange(this.last,  this.lastActive-1);
58543                     this.grid.getView().focusRow(this.lastActive);
58544                     if(last !== false){
58545                         this.last = last;
58546                     }
58547                 }else{
58548                     this.selectFirstRow();
58549                 }
58550                 this.fireEvent("afterselectionchange", this);
58551             },
58552             "down" : function(e){
58553                 if(!e.shiftKey){
58554                     this.selectNext(e.shiftKey);
58555                 }else if(this.last !== false && this.lastActive !== false){
58556                     var last = this.last;
58557                     this.selectRange(this.last,  this.lastActive+1);
58558                     this.grid.getView().focusRow(this.lastActive);
58559                     if(last !== false){
58560                         this.last = last;
58561                     }
58562                 }else{
58563                     this.selectFirstRow();
58564                 }
58565                 this.fireEvent("afterselectionchange", this);
58566             },
58567             scope: this
58568         });
58569
58570         var view = this.grid.view;
58571         view.on("refresh", this.onRefresh, this);
58572         view.on("rowupdated", this.onRowUpdated, this);
58573         view.on("rowremoved", this.onRemove, this);
58574     },
58575
58576     // private
58577     onRefresh : function(){
58578         var ds = this.grid.dataSource, i, v = this.grid.view;
58579         var s = this.selections;
58580         s.each(function(r){
58581             if((i = ds.indexOfId(r.id)) != -1){
58582                 v.onRowSelect(i);
58583                 s.add(ds.getAt(i)); // updating the selection relate data
58584             }else{
58585                 s.remove(r);
58586             }
58587         });
58588     },
58589
58590     // private
58591     onRemove : function(v, index, r){
58592         this.selections.remove(r);
58593     },
58594
58595     // private
58596     onRowUpdated : function(v, index, r){
58597         if(this.isSelected(r)){
58598             v.onRowSelect(index);
58599         }
58600     },
58601
58602     /**
58603      * Select records.
58604      * @param {Array} records The records to select
58605      * @param {Boolean} keepExisting (optional) True to keep existing selections
58606      */
58607     selectRecords : function(records, keepExisting){
58608         if(!keepExisting){
58609             this.clearSelections();
58610         }
58611         var ds = this.grid.dataSource;
58612         for(var i = 0, len = records.length; i < len; i++){
58613             this.selectRow(ds.indexOf(records[i]), true);
58614         }
58615     },
58616
58617     /**
58618      * Gets the number of selected rows.
58619      * @return {Number}
58620      */
58621     getCount : function(){
58622         return this.selections.length;
58623     },
58624
58625     /**
58626      * Selects the first row in the grid.
58627      */
58628     selectFirstRow : function(){
58629         this.selectRow(0);
58630     },
58631
58632     /**
58633      * Select the last row.
58634      * @param {Boolean} keepExisting (optional) True to keep existing selections
58635      */
58636     selectLastRow : function(keepExisting){
58637         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58638     },
58639
58640     /**
58641      * Selects the row immediately following the last selected row.
58642      * @param {Boolean} keepExisting (optional) True to keep existing selections
58643      */
58644     selectNext : function(keepExisting){
58645         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58646             this.selectRow(this.last+1, keepExisting);
58647             this.grid.getView().focusRow(this.last);
58648         }
58649     },
58650
58651     /**
58652      * Selects the row that precedes the last selected row.
58653      * @param {Boolean} keepExisting (optional) True to keep existing selections
58654      */
58655     selectPrevious : function(keepExisting){
58656         if(this.last){
58657             this.selectRow(this.last-1, keepExisting);
58658             this.grid.getView().focusRow(this.last);
58659         }
58660     },
58661
58662     /**
58663      * Returns the selected records
58664      * @return {Array} Array of selected records
58665      */
58666     getSelections : function(){
58667         return [].concat(this.selections.items);
58668     },
58669
58670     /**
58671      * Returns the first selected record.
58672      * @return {Record}
58673      */
58674     getSelected : function(){
58675         return this.selections.itemAt(0);
58676     },
58677
58678
58679     /**
58680      * Clears all selections.
58681      */
58682     clearSelections : function(fast){
58683         if(this.locked) {
58684             return;
58685         }
58686         if(fast !== true){
58687             var ds = this.grid.dataSource;
58688             var s = this.selections;
58689             s.each(function(r){
58690                 this.deselectRow(ds.indexOfId(r.id));
58691             }, this);
58692             s.clear();
58693         }else{
58694             this.selections.clear();
58695         }
58696         this.last = false;
58697     },
58698
58699
58700     /**
58701      * Selects all rows.
58702      */
58703     selectAll : function(){
58704         if(this.locked) {
58705             return;
58706         }
58707         this.selections.clear();
58708         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58709             this.selectRow(i, true);
58710         }
58711     },
58712
58713     /**
58714      * Returns True if there is a selection.
58715      * @return {Boolean}
58716      */
58717     hasSelection : function(){
58718         return this.selections.length > 0;
58719     },
58720
58721     /**
58722      * Returns True if the specified row is selected.
58723      * @param {Number/Record} record The record or index of the record to check
58724      * @return {Boolean}
58725      */
58726     isSelected : function(index){
58727         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58728         return (r && this.selections.key(r.id) ? true : false);
58729     },
58730
58731     /**
58732      * Returns True if the specified record id is selected.
58733      * @param {String} id The id of record to check
58734      * @return {Boolean}
58735      */
58736     isIdSelected : function(id){
58737         return (this.selections.key(id) ? true : false);
58738     },
58739
58740     // private
58741     handleMouseDown : function(e, t){
58742         var view = this.grid.getView(), rowIndex;
58743         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58744             return;
58745         };
58746         if(e.shiftKey && this.last !== false){
58747             var last = this.last;
58748             this.selectRange(last, rowIndex, e.ctrlKey);
58749             this.last = last; // reset the last
58750             view.focusRow(rowIndex);
58751         }else{
58752             var isSelected = this.isSelected(rowIndex);
58753             if(e.button !== 0 && isSelected){
58754                 view.focusRow(rowIndex);
58755             }else if(e.ctrlKey && isSelected){
58756                 this.deselectRow(rowIndex);
58757             }else if(!isSelected){
58758                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58759                 view.focusRow(rowIndex);
58760             }
58761         }
58762         this.fireEvent("afterselectionchange", this);
58763     },
58764     // private
58765     handleDragableRowClick :  function(grid, rowIndex, e) 
58766     {
58767         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58768             this.selectRow(rowIndex, false);
58769             grid.view.focusRow(rowIndex);
58770              this.fireEvent("afterselectionchange", this);
58771         }
58772     },
58773     
58774     /**
58775      * Selects multiple rows.
58776      * @param {Array} rows Array of the indexes of the row to select
58777      * @param {Boolean} keepExisting (optional) True to keep existing selections
58778      */
58779     selectRows : function(rows, keepExisting){
58780         if(!keepExisting){
58781             this.clearSelections();
58782         }
58783         for(var i = 0, len = rows.length; i < len; i++){
58784             this.selectRow(rows[i], true);
58785         }
58786     },
58787
58788     /**
58789      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58790      * @param {Number} startRow The index of the first row in the range
58791      * @param {Number} endRow The index of the last row in the range
58792      * @param {Boolean} keepExisting (optional) True to retain existing selections
58793      */
58794     selectRange : function(startRow, endRow, keepExisting){
58795         if(this.locked) {
58796             return;
58797         }
58798         if(!keepExisting){
58799             this.clearSelections();
58800         }
58801         if(startRow <= endRow){
58802             for(var i = startRow; i <= endRow; i++){
58803                 this.selectRow(i, true);
58804             }
58805         }else{
58806             for(var i = startRow; i >= endRow; i--){
58807                 this.selectRow(i, true);
58808             }
58809         }
58810     },
58811
58812     /**
58813      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58814      * @param {Number} startRow The index of the first row in the range
58815      * @param {Number} endRow The index of the last row in the range
58816      */
58817     deselectRange : function(startRow, endRow, preventViewNotify){
58818         if(this.locked) {
58819             return;
58820         }
58821         for(var i = startRow; i <= endRow; i++){
58822             this.deselectRow(i, preventViewNotify);
58823         }
58824     },
58825
58826     /**
58827      * Selects a row.
58828      * @param {Number} row The index of the row to select
58829      * @param {Boolean} keepExisting (optional) True to keep existing selections
58830      */
58831     selectRow : function(index, keepExisting, preventViewNotify){
58832         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58833             return;
58834         }
58835         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58836             if(!keepExisting || this.singleSelect){
58837                 this.clearSelections();
58838             }
58839             var r = this.grid.dataSource.getAt(index);
58840             this.selections.add(r);
58841             this.last = this.lastActive = index;
58842             if(!preventViewNotify){
58843                 this.grid.getView().onRowSelect(index);
58844             }
58845             this.fireEvent("rowselect", this, index, r);
58846             this.fireEvent("selectionchange", this);
58847         }
58848     },
58849
58850     /**
58851      * Deselects a row.
58852      * @param {Number} row The index of the row to deselect
58853      */
58854     deselectRow : function(index, preventViewNotify){
58855         if(this.locked) {
58856             return;
58857         }
58858         if(this.last == index){
58859             this.last = false;
58860         }
58861         if(this.lastActive == index){
58862             this.lastActive = false;
58863         }
58864         var r = this.grid.dataSource.getAt(index);
58865         this.selections.remove(r);
58866         if(!preventViewNotify){
58867             this.grid.getView().onRowDeselect(index);
58868         }
58869         this.fireEvent("rowdeselect", this, index);
58870         this.fireEvent("selectionchange", this);
58871     },
58872
58873     // private
58874     restoreLast : function(){
58875         if(this._last){
58876             this.last = this._last;
58877         }
58878     },
58879
58880     // private
58881     acceptsNav : function(row, col, cm){
58882         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58883     },
58884
58885     // private
58886     onEditorKey : function(field, e){
58887         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58888         if(k == e.TAB){
58889             e.stopEvent();
58890             ed.completeEdit();
58891             if(e.shiftKey){
58892                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58893             }else{
58894                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58895             }
58896         }else if(k == e.ENTER && !e.ctrlKey){
58897             e.stopEvent();
58898             ed.completeEdit();
58899             if(e.shiftKey){
58900                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58901             }else{
58902                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58903             }
58904         }else if(k == e.ESC){
58905             ed.cancelEdit();
58906         }
58907         if(newCell){
58908             g.startEditing(newCell[0], newCell[1]);
58909         }
58910     }
58911 });/*
58912  * Based on:
58913  * Ext JS Library 1.1.1
58914  * Copyright(c) 2006-2007, Ext JS, LLC.
58915  *
58916  * Originally Released Under LGPL - original licence link has changed is not relivant.
58917  *
58918  * Fork - LGPL
58919  * <script type="text/javascript">
58920  */
58921 /**
58922  * @class Roo.grid.CellSelectionModel
58923  * @extends Roo.grid.AbstractSelectionModel
58924  * This class provides the basic implementation for cell selection in a grid.
58925  * @constructor
58926  * @param {Object} config The object containing the configuration of this model.
58927  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58928  */
58929 Roo.grid.CellSelectionModel = function(config){
58930     Roo.apply(this, config);
58931
58932     this.selection = null;
58933
58934     this.addEvents({
58935         /**
58936              * @event beforerowselect
58937              * Fires before a cell is selected.
58938              * @param {SelectionModel} this
58939              * @param {Number} rowIndex The selected row index
58940              * @param {Number} colIndex The selected cell index
58941              */
58942             "beforecellselect" : true,
58943         /**
58944              * @event cellselect
58945              * Fires when a cell is selected.
58946              * @param {SelectionModel} this
58947              * @param {Number} rowIndex The selected row index
58948              * @param {Number} colIndex The selected cell index
58949              */
58950             "cellselect" : true,
58951         /**
58952              * @event selectionchange
58953              * Fires when the active selection changes.
58954              * @param {SelectionModel} this
58955              * @param {Object} selection null for no selection or an object (o) with two properties
58956                 <ul>
58957                 <li>o.record: the record object for the row the selection is in</li>
58958                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58959                 </ul>
58960              */
58961             "selectionchange" : true,
58962         /**
58963              * @event tabend
58964              * Fires when the tab (or enter) was pressed on the last editable cell
58965              * You can use this to trigger add new row.
58966              * @param {SelectionModel} this
58967              */
58968             "tabend" : true,
58969          /**
58970              * @event beforeeditnext
58971              * Fires before the next editable sell is made active
58972              * You can use this to skip to another cell or fire the tabend
58973              *    if you set cell to false
58974              * @param {Object} eventdata object : { cell : [ row, col ] } 
58975              */
58976             "beforeeditnext" : true
58977     });
58978     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58979 };
58980
58981 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58982     
58983     enter_is_tab: false,
58984
58985     /** @ignore */
58986     initEvents : function(){
58987         this.grid.on("mousedown", this.handleMouseDown, this);
58988         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58989         var view = this.grid.view;
58990         view.on("refresh", this.onViewChange, this);
58991         view.on("rowupdated", this.onRowUpdated, this);
58992         view.on("beforerowremoved", this.clearSelections, this);
58993         view.on("beforerowsinserted", this.clearSelections, this);
58994         if(this.grid.isEditor){
58995             this.grid.on("beforeedit", this.beforeEdit,  this);
58996         }
58997     },
58998
58999         //private
59000     beforeEdit : function(e){
59001         this.select(e.row, e.column, false, true, e.record);
59002     },
59003
59004         //private
59005     onRowUpdated : function(v, index, r){
59006         if(this.selection && this.selection.record == r){
59007             v.onCellSelect(index, this.selection.cell[1]);
59008         }
59009     },
59010
59011         //private
59012     onViewChange : function(){
59013         this.clearSelections(true);
59014     },
59015
59016         /**
59017          * Returns the currently selected cell,.
59018          * @return {Array} The selected cell (row, column) or null if none selected.
59019          */
59020     getSelectedCell : function(){
59021         return this.selection ? this.selection.cell : null;
59022     },
59023
59024     /**
59025      * Clears all selections.
59026      * @param {Boolean} true to prevent the gridview from being notified about the change.
59027      */
59028     clearSelections : function(preventNotify){
59029         var s = this.selection;
59030         if(s){
59031             if(preventNotify !== true){
59032                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
59033             }
59034             this.selection = null;
59035             this.fireEvent("selectionchange", this, null);
59036         }
59037     },
59038
59039     /**
59040      * Returns true if there is a selection.
59041      * @return {Boolean}
59042      */
59043     hasSelection : function(){
59044         return this.selection ? true : false;
59045     },
59046
59047     /** @ignore */
59048     handleMouseDown : function(e, t){
59049         var v = this.grid.getView();
59050         if(this.isLocked()){
59051             return;
59052         };
59053         var row = v.findRowIndex(t);
59054         var cell = v.findCellIndex(t);
59055         if(row !== false && cell !== false){
59056             this.select(row, cell);
59057         }
59058     },
59059
59060     /**
59061      * Selects a cell.
59062      * @param {Number} rowIndex
59063      * @param {Number} collIndex
59064      */
59065     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59066         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59067             this.clearSelections();
59068             r = r || this.grid.dataSource.getAt(rowIndex);
59069             this.selection = {
59070                 record : r,
59071                 cell : [rowIndex, colIndex]
59072             };
59073             if(!preventViewNotify){
59074                 var v = this.grid.getView();
59075                 v.onCellSelect(rowIndex, colIndex);
59076                 if(preventFocus !== true){
59077                     v.focusCell(rowIndex, colIndex);
59078                 }
59079             }
59080             this.fireEvent("cellselect", this, rowIndex, colIndex);
59081             this.fireEvent("selectionchange", this, this.selection);
59082         }
59083     },
59084
59085         //private
59086     isSelectable : function(rowIndex, colIndex, cm){
59087         return !cm.isHidden(colIndex);
59088     },
59089
59090     /** @ignore */
59091     handleKeyDown : function(e){
59092         //Roo.log('Cell Sel Model handleKeyDown');
59093         if(!e.isNavKeyPress()){
59094             return;
59095         }
59096         var g = this.grid, s = this.selection;
59097         if(!s){
59098             e.stopEvent();
59099             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59100             if(cell){
59101                 this.select(cell[0], cell[1]);
59102             }
59103             return;
59104         }
59105         var sm = this;
59106         var walk = function(row, col, step){
59107             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59108         };
59109         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59110         var newCell;
59111
59112       
59113
59114         switch(k){
59115             case e.TAB:
59116                 // handled by onEditorKey
59117                 if (g.isEditor && g.editing) {
59118                     return;
59119                 }
59120                 if(e.shiftKey) {
59121                     newCell = walk(r, c-1, -1);
59122                 } else {
59123                     newCell = walk(r, c+1, 1);
59124                 }
59125                 break;
59126             
59127             case e.DOWN:
59128                newCell = walk(r+1, c, 1);
59129                 break;
59130             
59131             case e.UP:
59132                 newCell = walk(r-1, c, -1);
59133                 break;
59134             
59135             case e.RIGHT:
59136                 newCell = walk(r, c+1, 1);
59137                 break;
59138             
59139             case e.LEFT:
59140                 newCell = walk(r, c-1, -1);
59141                 break;
59142             
59143             case e.ENTER:
59144                 
59145                 if(g.isEditor && !g.editing){
59146                    g.startEditing(r, c);
59147                    e.stopEvent();
59148                    return;
59149                 }
59150                 
59151                 
59152              break;
59153         };
59154         if(newCell){
59155             this.select(newCell[0], newCell[1]);
59156             e.stopEvent();
59157             
59158         }
59159     },
59160
59161     acceptsNav : function(row, col, cm){
59162         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59163     },
59164     /**
59165      * Selects a cell.
59166      * @param {Number} field (not used) - as it's normally used as a listener
59167      * @param {Number} e - event - fake it by using
59168      *
59169      * var e = Roo.EventObjectImpl.prototype;
59170      * e.keyCode = e.TAB
59171      *
59172      * 
59173      */
59174     onEditorKey : function(field, e){
59175         
59176         var k = e.getKey(),
59177             newCell,
59178             g = this.grid,
59179             ed = g.activeEditor,
59180             forward = false;
59181         ///Roo.log('onEditorKey' + k);
59182         
59183         
59184         if (this.enter_is_tab && k == e.ENTER) {
59185             k = e.TAB;
59186         }
59187         
59188         if(k == e.TAB){
59189             if(e.shiftKey){
59190                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59191             }else{
59192                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59193                 forward = true;
59194             }
59195             
59196             e.stopEvent();
59197             
59198         } else if(k == e.ENTER &&  !e.ctrlKey){
59199             ed.completeEdit();
59200             e.stopEvent();
59201             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59202         
59203                 } else if(k == e.ESC){
59204             ed.cancelEdit();
59205         }
59206                 
59207         if (newCell) {
59208             var ecall = { cell : newCell, forward : forward };
59209             this.fireEvent('beforeeditnext', ecall );
59210             newCell = ecall.cell;
59211                         forward = ecall.forward;
59212         }
59213                 
59214         if(newCell){
59215             //Roo.log('next cell after edit');
59216             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59217         } else if (forward) {
59218             // tabbed past last
59219             this.fireEvent.defer(100, this, ['tabend',this]);
59220         }
59221     }
59222 });/*
59223  * Based on:
59224  * Ext JS Library 1.1.1
59225  * Copyright(c) 2006-2007, Ext JS, LLC.
59226  *
59227  * Originally Released Under LGPL - original licence link has changed is not relivant.
59228  *
59229  * Fork - LGPL
59230  * <script type="text/javascript">
59231  */
59232  
59233 /**
59234  * @class Roo.grid.EditorGrid
59235  * @extends Roo.grid.Grid
59236  * Class for creating and editable grid.
59237  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59238  * The container MUST have some type of size defined for the grid to fill. The container will be 
59239  * automatically set to position relative if it isn't already.
59240  * @param {Object} dataSource The data model to bind to
59241  * @param {Object} colModel The column model with info about this grid's columns
59242  */
59243 Roo.grid.EditorGrid = function(container, config){
59244     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59245     this.getGridEl().addClass("xedit-grid");
59246
59247     if(!this.selModel){
59248         this.selModel = new Roo.grid.CellSelectionModel();
59249     }
59250
59251     this.activeEditor = null;
59252
59253         this.addEvents({
59254             /**
59255              * @event beforeedit
59256              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59257              * <ul style="padding:5px;padding-left:16px;">
59258              * <li>grid - This grid</li>
59259              * <li>record - The record being edited</li>
59260              * <li>field - The field name being edited</li>
59261              * <li>value - The value for the field being edited.</li>
59262              * <li>row - The grid row index</li>
59263              * <li>column - The grid column index</li>
59264              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59265              * </ul>
59266              * @param {Object} e An edit event (see above for description)
59267              */
59268             "beforeedit" : true,
59269             /**
59270              * @event afteredit
59271              * Fires after a cell is edited. <br />
59272              * <ul style="padding:5px;padding-left:16px;">
59273              * <li>grid - This grid</li>
59274              * <li>record - The record being edited</li>
59275              * <li>field - The field name being edited</li>
59276              * <li>value - The value being set</li>
59277              * <li>originalValue - The original value for the field, before the edit.</li>
59278              * <li>row - The grid row index</li>
59279              * <li>column - The grid column index</li>
59280              * </ul>
59281              * @param {Object} e An edit event (see above for description)
59282              */
59283             "afteredit" : true,
59284             /**
59285              * @event validateedit
59286              * Fires after a cell is edited, but before the value is set in the record. 
59287          * You can use this to modify the value being set in the field, Return false
59288              * to cancel the change. The edit event object has the following properties <br />
59289              * <ul style="padding:5px;padding-left:16px;">
59290          * <li>editor - This editor</li>
59291              * <li>grid - This grid</li>
59292              * <li>record - The record being edited</li>
59293              * <li>field - The field name being edited</li>
59294              * <li>value - The value being set</li>
59295              * <li>originalValue - The original value for the field, before the edit.</li>
59296              * <li>row - The grid row index</li>
59297              * <li>column - The grid column index</li>
59298              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59299              * </ul>
59300              * @param {Object} e An edit event (see above for description)
59301              */
59302             "validateedit" : true
59303         });
59304     this.on("bodyscroll", this.stopEditing,  this);
59305     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59306 };
59307
59308 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59309     /**
59310      * @cfg {Number} clicksToEdit
59311      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59312      */
59313     clicksToEdit: 2,
59314
59315     // private
59316     isEditor : true,
59317     // private
59318     trackMouseOver: false, // causes very odd FF errors
59319
59320     onCellDblClick : function(g, row, col){
59321         this.startEditing(row, col);
59322     },
59323
59324     onEditComplete : function(ed, value, startValue){
59325         this.editing = false;
59326         this.activeEditor = null;
59327         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59328         var r = ed.record;
59329         var field = this.colModel.getDataIndex(ed.col);
59330         var e = {
59331             grid: this,
59332             record: r,
59333             field: field,
59334             originalValue: startValue,
59335             value: value,
59336             row: ed.row,
59337             column: ed.col,
59338             cancel:false,
59339             editor: ed
59340         };
59341         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59342         cell.show();
59343           
59344         if(String(value) !== String(startValue)){
59345             
59346             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59347                 r.set(field, e.value);
59348                 // if we are dealing with a combo box..
59349                 // then we also set the 'name' colum to be the displayField
59350                 if (ed.field.displayField && ed.field.name) {
59351                     r.set(ed.field.name, ed.field.el.dom.value);
59352                 }
59353                 
59354                 delete e.cancel; //?? why!!!
59355                 this.fireEvent("afteredit", e);
59356             }
59357         } else {
59358             this.fireEvent("afteredit", e); // always fire it!
59359         }
59360         this.view.focusCell(ed.row, ed.col);
59361     },
59362
59363     /**
59364      * Starts editing the specified for the specified row/column
59365      * @param {Number} rowIndex
59366      * @param {Number} colIndex
59367      */
59368     startEditing : function(row, col){
59369         this.stopEditing();
59370         if(this.colModel.isCellEditable(col, row)){
59371             this.view.ensureVisible(row, col, true);
59372           
59373             var r = this.dataSource.getAt(row);
59374             var field = this.colModel.getDataIndex(col);
59375             var cell = Roo.get(this.view.getCell(row,col));
59376             var e = {
59377                 grid: this,
59378                 record: r,
59379                 field: field,
59380                 value: r.data[field],
59381                 row: row,
59382                 column: col,
59383                 cancel:false 
59384             };
59385             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59386                 this.editing = true;
59387                 var ed = this.colModel.getCellEditor(col, row);
59388                 
59389                 if (!ed) {
59390                     return;
59391                 }
59392                 if(!ed.rendered){
59393                     ed.render(ed.parentEl || document.body);
59394                 }
59395                 ed.field.reset();
59396                
59397                 cell.hide();
59398                 
59399                 (function(){ // complex but required for focus issues in safari, ie and opera
59400                     ed.row = row;
59401                     ed.col = col;
59402                     ed.record = r;
59403                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59404                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59405                     this.activeEditor = ed;
59406                     var v = r.data[field];
59407                     ed.startEdit(this.view.getCell(row, col), v);
59408                     // combo's with 'displayField and name set
59409                     if (ed.field.displayField && ed.field.name) {
59410                         ed.field.el.dom.value = r.data[ed.field.name];
59411                     }
59412                     
59413                     
59414                 }).defer(50, this);
59415             }
59416         }
59417     },
59418         
59419     /**
59420      * Stops any active editing
59421      */
59422     stopEditing : function(){
59423         if(this.activeEditor){
59424             this.activeEditor.completeEdit();
59425         }
59426         this.activeEditor = null;
59427     },
59428         
59429          /**
59430      * Called to get grid's drag proxy text, by default returns this.ddText.
59431      * @return {String}
59432      */
59433     getDragDropText : function(){
59434         var count = this.selModel.getSelectedCell() ? 1 : 0;
59435         return String.format(this.ddText, count, count == 1 ? '' : 's');
59436     }
59437         
59438 });/*
59439  * Based on:
59440  * Ext JS Library 1.1.1
59441  * Copyright(c) 2006-2007, Ext JS, LLC.
59442  *
59443  * Originally Released Under LGPL - original licence link has changed is not relivant.
59444  *
59445  * Fork - LGPL
59446  * <script type="text/javascript">
59447  */
59448
59449 // private - not really -- you end up using it !
59450 // This is a support class used internally by the Grid components
59451
59452 /**
59453  * @class Roo.grid.GridEditor
59454  * @extends Roo.Editor
59455  * Class for creating and editable grid elements.
59456  * @param {Object} config any settings (must include field)
59457  */
59458 Roo.grid.GridEditor = function(field, config){
59459     if (!config && field.field) {
59460         config = field;
59461         field = Roo.factory(config.field, Roo.form);
59462     }
59463     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59464     field.monitorTab = false;
59465 };
59466
59467 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59468     
59469     /**
59470      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59471      */
59472     
59473     alignment: "tl-tl",
59474     autoSize: "width",
59475     hideEl : false,
59476     cls: "x-small-editor x-grid-editor",
59477     shim:false,
59478     shadow:"frame"
59479 });/*
59480  * Based on:
59481  * Ext JS Library 1.1.1
59482  * Copyright(c) 2006-2007, Ext JS, LLC.
59483  *
59484  * Originally Released Under LGPL - original licence link has changed is not relivant.
59485  *
59486  * Fork - LGPL
59487  * <script type="text/javascript">
59488  */
59489   
59490
59491   
59492 Roo.grid.PropertyRecord = Roo.data.Record.create([
59493     {name:'name',type:'string'},  'value'
59494 ]);
59495
59496
59497 Roo.grid.PropertyStore = function(grid, source){
59498     this.grid = grid;
59499     this.store = new Roo.data.Store({
59500         recordType : Roo.grid.PropertyRecord
59501     });
59502     this.store.on('update', this.onUpdate,  this);
59503     if(source){
59504         this.setSource(source);
59505     }
59506     Roo.grid.PropertyStore.superclass.constructor.call(this);
59507 };
59508
59509
59510
59511 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59512     setSource : function(o){
59513         this.source = o;
59514         this.store.removeAll();
59515         var data = [];
59516         for(var k in o){
59517             if(this.isEditableValue(o[k])){
59518                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59519             }
59520         }
59521         this.store.loadRecords({records: data}, {}, true);
59522     },
59523
59524     onUpdate : function(ds, record, type){
59525         if(type == Roo.data.Record.EDIT){
59526             var v = record.data['value'];
59527             var oldValue = record.modified['value'];
59528             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59529                 this.source[record.id] = v;
59530                 record.commit();
59531                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59532             }else{
59533                 record.reject();
59534             }
59535         }
59536     },
59537
59538     getProperty : function(row){
59539        return this.store.getAt(row);
59540     },
59541
59542     isEditableValue: function(val){
59543         if(val && val instanceof Date){
59544             return true;
59545         }else if(typeof val == 'object' || typeof val == 'function'){
59546             return false;
59547         }
59548         return true;
59549     },
59550
59551     setValue : function(prop, value){
59552         this.source[prop] = value;
59553         this.store.getById(prop).set('value', value);
59554     },
59555
59556     getSource : function(){
59557         return this.source;
59558     }
59559 });
59560
59561 Roo.grid.PropertyColumnModel = function(grid, store){
59562     this.grid = grid;
59563     var g = Roo.grid;
59564     g.PropertyColumnModel.superclass.constructor.call(this, [
59565         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59566         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59567     ]);
59568     this.store = store;
59569     this.bselect = Roo.DomHelper.append(document.body, {
59570         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59571             {tag: 'option', value: 'true', html: 'true'},
59572             {tag: 'option', value: 'false', html: 'false'}
59573         ]
59574     });
59575     Roo.id(this.bselect);
59576     var f = Roo.form;
59577     this.editors = {
59578         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59579         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59580         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59581         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59582         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59583     };
59584     this.renderCellDelegate = this.renderCell.createDelegate(this);
59585     this.renderPropDelegate = this.renderProp.createDelegate(this);
59586 };
59587
59588 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59589     
59590     
59591     nameText : 'Name',
59592     valueText : 'Value',
59593     
59594     dateFormat : 'm/j/Y',
59595     
59596     
59597     renderDate : function(dateVal){
59598         return dateVal.dateFormat(this.dateFormat);
59599     },
59600
59601     renderBool : function(bVal){
59602         return bVal ? 'true' : 'false';
59603     },
59604
59605     isCellEditable : function(colIndex, rowIndex){
59606         return colIndex == 1;
59607     },
59608
59609     getRenderer : function(col){
59610         return col == 1 ?
59611             this.renderCellDelegate : this.renderPropDelegate;
59612     },
59613
59614     renderProp : function(v){
59615         return this.getPropertyName(v);
59616     },
59617
59618     renderCell : function(val){
59619         var rv = val;
59620         if(val instanceof Date){
59621             rv = this.renderDate(val);
59622         }else if(typeof val == 'boolean'){
59623             rv = this.renderBool(val);
59624         }
59625         return Roo.util.Format.htmlEncode(rv);
59626     },
59627
59628     getPropertyName : function(name){
59629         var pn = this.grid.propertyNames;
59630         return pn && pn[name] ? pn[name] : name;
59631     },
59632
59633     getCellEditor : function(colIndex, rowIndex){
59634         var p = this.store.getProperty(rowIndex);
59635         var n = p.data['name'], val = p.data['value'];
59636         
59637         if(typeof(this.grid.customEditors[n]) == 'string'){
59638             return this.editors[this.grid.customEditors[n]];
59639         }
59640         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59641             return this.grid.customEditors[n];
59642         }
59643         if(val instanceof Date){
59644             return this.editors['date'];
59645         }else if(typeof val == 'number'){
59646             return this.editors['number'];
59647         }else if(typeof val == 'boolean'){
59648             return this.editors['boolean'];
59649         }else{
59650             return this.editors['string'];
59651         }
59652     }
59653 });
59654
59655 /**
59656  * @class Roo.grid.PropertyGrid
59657  * @extends Roo.grid.EditorGrid
59658  * This class represents the  interface of a component based property grid control.
59659  * <br><br>Usage:<pre><code>
59660  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59661       
59662  });
59663  // set any options
59664  grid.render();
59665  * </code></pre>
59666   
59667  * @constructor
59668  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59669  * The container MUST have some type of size defined for the grid to fill. The container will be
59670  * automatically set to position relative if it isn't already.
59671  * @param {Object} config A config object that sets properties on this grid.
59672  */
59673 Roo.grid.PropertyGrid = function(container, config){
59674     config = config || {};
59675     var store = new Roo.grid.PropertyStore(this);
59676     this.store = store;
59677     var cm = new Roo.grid.PropertyColumnModel(this, store);
59678     store.store.sort('name', 'ASC');
59679     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59680         ds: store.store,
59681         cm: cm,
59682         enableColLock:false,
59683         enableColumnMove:false,
59684         stripeRows:false,
59685         trackMouseOver: false,
59686         clicksToEdit:1
59687     }, config));
59688     this.getGridEl().addClass('x-props-grid');
59689     this.lastEditRow = null;
59690     this.on('columnresize', this.onColumnResize, this);
59691     this.addEvents({
59692          /**
59693              * @event beforepropertychange
59694              * Fires before a property changes (return false to stop?)
59695              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59696              * @param {String} id Record Id
59697              * @param {String} newval New Value
59698          * @param {String} oldval Old Value
59699              */
59700         "beforepropertychange": true,
59701         /**
59702              * @event propertychange
59703              * Fires after a property changes
59704              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59705              * @param {String} id Record Id
59706              * @param {String} newval New Value
59707          * @param {String} oldval Old Value
59708              */
59709         "propertychange": true
59710     });
59711     this.customEditors = this.customEditors || {};
59712 };
59713 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59714     
59715      /**
59716      * @cfg {Object} customEditors map of colnames=> custom editors.
59717      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59718      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59719      * false disables editing of the field.
59720          */
59721     
59722       /**
59723      * @cfg {Object} propertyNames map of property Names to their displayed value
59724          */
59725     
59726     render : function(){
59727         Roo.grid.PropertyGrid.superclass.render.call(this);
59728         this.autoSize.defer(100, this);
59729     },
59730
59731     autoSize : function(){
59732         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59733         if(this.view){
59734             this.view.fitColumns();
59735         }
59736     },
59737
59738     onColumnResize : function(){
59739         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59740         this.autoSize();
59741     },
59742     /**
59743      * Sets the data for the Grid
59744      * accepts a Key => Value object of all the elements avaiable.
59745      * @param {Object} data  to appear in grid.
59746      */
59747     setSource : function(source){
59748         this.store.setSource(source);
59749         //this.autoSize();
59750     },
59751     /**
59752      * Gets all the data from the grid.
59753      * @return {Object} data  data stored in grid
59754      */
59755     getSource : function(){
59756         return this.store.getSource();
59757     }
59758 });/*
59759   
59760  * Licence LGPL
59761  
59762  */
59763  
59764 /**
59765  * @class Roo.grid.Calendar
59766  * @extends Roo.util.Grid
59767  * This class extends the Grid to provide a calendar widget
59768  * <br><br>Usage:<pre><code>
59769  var grid = new Roo.grid.Calendar("my-container-id", {
59770      ds: myDataStore,
59771      cm: myColModel,
59772      selModel: mySelectionModel,
59773      autoSizeColumns: true,
59774      monitorWindowResize: false,
59775      trackMouseOver: true
59776      eventstore : real data store..
59777  });
59778  // set any options
59779  grid.render();
59780   
59781   * @constructor
59782  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59783  * The container MUST have some type of size defined for the grid to fill. The container will be
59784  * automatically set to position relative if it isn't already.
59785  * @param {Object} config A config object that sets properties on this grid.
59786  */
59787 Roo.grid.Calendar = function(container, config){
59788         // initialize the container
59789         this.container = Roo.get(container);
59790         this.container.update("");
59791         this.container.setStyle("overflow", "hidden");
59792     this.container.addClass('x-grid-container');
59793
59794     this.id = this.container.id;
59795
59796     Roo.apply(this, config);
59797     // check and correct shorthanded configs
59798     
59799     var rows = [];
59800     var d =1;
59801     for (var r = 0;r < 6;r++) {
59802         
59803         rows[r]=[];
59804         for (var c =0;c < 7;c++) {
59805             rows[r][c]= '';
59806         }
59807     }
59808     if (this.eventStore) {
59809         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59810         this.eventStore.on('load',this.onLoad, this);
59811         this.eventStore.on('beforeload',this.clearEvents, this);
59812          
59813     }
59814     
59815     this.dataSource = new Roo.data.Store({
59816             proxy: new Roo.data.MemoryProxy(rows),
59817             reader: new Roo.data.ArrayReader({}, [
59818                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59819     });
59820
59821     this.dataSource.load();
59822     this.ds = this.dataSource;
59823     this.ds.xmodule = this.xmodule || false;
59824     
59825     
59826     var cellRender = function(v,x,r)
59827     {
59828         return String.format(
59829             '<div class="fc-day  fc-widget-content"><div>' +
59830                 '<div class="fc-event-container"></div>' +
59831                 '<div class="fc-day-number">{0}</div>'+
59832                 
59833                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59834             '</div></div>', v);
59835     
59836     }
59837     
59838     
59839     this.colModel = new Roo.grid.ColumnModel( [
59840         {
59841             xtype: 'ColumnModel',
59842             xns: Roo.grid,
59843             dataIndex : 'weekday0',
59844             header : 'Sunday',
59845             renderer : cellRender
59846         },
59847         {
59848             xtype: 'ColumnModel',
59849             xns: Roo.grid,
59850             dataIndex : 'weekday1',
59851             header : 'Monday',
59852             renderer : cellRender
59853         },
59854         {
59855             xtype: 'ColumnModel',
59856             xns: Roo.grid,
59857             dataIndex : 'weekday2',
59858             header : 'Tuesday',
59859             renderer : cellRender
59860         },
59861         {
59862             xtype: 'ColumnModel',
59863             xns: Roo.grid,
59864             dataIndex : 'weekday3',
59865             header : 'Wednesday',
59866             renderer : cellRender
59867         },
59868         {
59869             xtype: 'ColumnModel',
59870             xns: Roo.grid,
59871             dataIndex : 'weekday4',
59872             header : 'Thursday',
59873             renderer : cellRender
59874         },
59875         {
59876             xtype: 'ColumnModel',
59877             xns: Roo.grid,
59878             dataIndex : 'weekday5',
59879             header : 'Friday',
59880             renderer : cellRender
59881         },
59882         {
59883             xtype: 'ColumnModel',
59884             xns: Roo.grid,
59885             dataIndex : 'weekday6',
59886             header : 'Saturday',
59887             renderer : cellRender
59888         }
59889     ]);
59890     this.cm = this.colModel;
59891     this.cm.xmodule = this.xmodule || false;
59892  
59893         
59894           
59895     //this.selModel = new Roo.grid.CellSelectionModel();
59896     //this.sm = this.selModel;
59897     //this.selModel.init(this);
59898     
59899     
59900     if(this.width){
59901         this.container.setWidth(this.width);
59902     }
59903
59904     if(this.height){
59905         this.container.setHeight(this.height);
59906     }
59907     /** @private */
59908         this.addEvents({
59909         // raw events
59910         /**
59911          * @event click
59912          * The raw click event for the entire grid.
59913          * @param {Roo.EventObject} e
59914          */
59915         "click" : true,
59916         /**
59917          * @event dblclick
59918          * The raw dblclick event for the entire grid.
59919          * @param {Roo.EventObject} e
59920          */
59921         "dblclick" : true,
59922         /**
59923          * @event contextmenu
59924          * The raw contextmenu event for the entire grid.
59925          * @param {Roo.EventObject} e
59926          */
59927         "contextmenu" : true,
59928         /**
59929          * @event mousedown
59930          * The raw mousedown event for the entire grid.
59931          * @param {Roo.EventObject} e
59932          */
59933         "mousedown" : true,
59934         /**
59935          * @event mouseup
59936          * The raw mouseup event for the entire grid.
59937          * @param {Roo.EventObject} e
59938          */
59939         "mouseup" : true,
59940         /**
59941          * @event mouseover
59942          * The raw mouseover event for the entire grid.
59943          * @param {Roo.EventObject} e
59944          */
59945         "mouseover" : true,
59946         /**
59947          * @event mouseout
59948          * The raw mouseout event for the entire grid.
59949          * @param {Roo.EventObject} e
59950          */
59951         "mouseout" : true,
59952         /**
59953          * @event keypress
59954          * The raw keypress event for the entire grid.
59955          * @param {Roo.EventObject} e
59956          */
59957         "keypress" : true,
59958         /**
59959          * @event keydown
59960          * The raw keydown event for the entire grid.
59961          * @param {Roo.EventObject} e
59962          */
59963         "keydown" : true,
59964
59965         // custom events
59966
59967         /**
59968          * @event cellclick
59969          * Fires when a cell is clicked
59970          * @param {Grid} this
59971          * @param {Number} rowIndex
59972          * @param {Number} columnIndex
59973          * @param {Roo.EventObject} e
59974          */
59975         "cellclick" : true,
59976         /**
59977          * @event celldblclick
59978          * Fires when a cell is double clicked
59979          * @param {Grid} this
59980          * @param {Number} rowIndex
59981          * @param {Number} columnIndex
59982          * @param {Roo.EventObject} e
59983          */
59984         "celldblclick" : true,
59985         /**
59986          * @event rowclick
59987          * Fires when a row is clicked
59988          * @param {Grid} this
59989          * @param {Number} rowIndex
59990          * @param {Roo.EventObject} e
59991          */
59992         "rowclick" : true,
59993         /**
59994          * @event rowdblclick
59995          * Fires when a row is double clicked
59996          * @param {Grid} this
59997          * @param {Number} rowIndex
59998          * @param {Roo.EventObject} e
59999          */
60000         "rowdblclick" : true,
60001         /**
60002          * @event headerclick
60003          * Fires when a header is clicked
60004          * @param {Grid} this
60005          * @param {Number} columnIndex
60006          * @param {Roo.EventObject} e
60007          */
60008         "headerclick" : true,
60009         /**
60010          * @event headerdblclick
60011          * Fires when a header cell is double clicked
60012          * @param {Grid} this
60013          * @param {Number} columnIndex
60014          * @param {Roo.EventObject} e
60015          */
60016         "headerdblclick" : true,
60017         /**
60018          * @event rowcontextmenu
60019          * Fires when a row is right clicked
60020          * @param {Grid} this
60021          * @param {Number} rowIndex
60022          * @param {Roo.EventObject} e
60023          */
60024         "rowcontextmenu" : true,
60025         /**
60026          * @event cellcontextmenu
60027          * Fires when a cell is right clicked
60028          * @param {Grid} this
60029          * @param {Number} rowIndex
60030          * @param {Number} cellIndex
60031          * @param {Roo.EventObject} e
60032          */
60033          "cellcontextmenu" : true,
60034         /**
60035          * @event headercontextmenu
60036          * Fires when a header is right clicked
60037          * @param {Grid} this
60038          * @param {Number} columnIndex
60039          * @param {Roo.EventObject} e
60040          */
60041         "headercontextmenu" : true,
60042         /**
60043          * @event bodyscroll
60044          * Fires when the body element is scrolled
60045          * @param {Number} scrollLeft
60046          * @param {Number} scrollTop
60047          */
60048         "bodyscroll" : true,
60049         /**
60050          * @event columnresize
60051          * Fires when the user resizes a column
60052          * @param {Number} columnIndex
60053          * @param {Number} newSize
60054          */
60055         "columnresize" : true,
60056         /**
60057          * @event columnmove
60058          * Fires when the user moves a column
60059          * @param {Number} oldIndex
60060          * @param {Number} newIndex
60061          */
60062         "columnmove" : true,
60063         /**
60064          * @event startdrag
60065          * Fires when row(s) start being dragged
60066          * @param {Grid} this
60067          * @param {Roo.GridDD} dd The drag drop object
60068          * @param {event} e The raw browser event
60069          */
60070         "startdrag" : true,
60071         /**
60072          * @event enddrag
60073          * Fires when a drag operation is complete
60074          * @param {Grid} this
60075          * @param {Roo.GridDD} dd The drag drop object
60076          * @param {event} e The raw browser event
60077          */
60078         "enddrag" : true,
60079         /**
60080          * @event dragdrop
60081          * Fires when dragged row(s) are dropped on a valid DD target
60082          * @param {Grid} this
60083          * @param {Roo.GridDD} dd The drag drop object
60084          * @param {String} targetId The target drag drop object
60085          * @param {event} e The raw browser event
60086          */
60087         "dragdrop" : true,
60088         /**
60089          * @event dragover
60090          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60091          * @param {Grid} this
60092          * @param {Roo.GridDD} dd The drag drop object
60093          * @param {String} targetId The target drag drop object
60094          * @param {event} e The raw browser event
60095          */
60096         "dragover" : true,
60097         /**
60098          * @event dragenter
60099          *  Fires when the dragged row(s) first cross another DD target while being dragged
60100          * @param {Grid} this
60101          * @param {Roo.GridDD} dd The drag drop object
60102          * @param {String} targetId The target drag drop object
60103          * @param {event} e The raw browser event
60104          */
60105         "dragenter" : true,
60106         /**
60107          * @event dragout
60108          * Fires when the dragged row(s) leave another DD target while being dragged
60109          * @param {Grid} this
60110          * @param {Roo.GridDD} dd The drag drop object
60111          * @param {String} targetId The target drag drop object
60112          * @param {event} e The raw browser event
60113          */
60114         "dragout" : true,
60115         /**
60116          * @event rowclass
60117          * Fires when a row is rendered, so you can change add a style to it.
60118          * @param {GridView} gridview   The grid view
60119          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60120          */
60121         'rowclass' : true,
60122
60123         /**
60124          * @event render
60125          * Fires when the grid is rendered
60126          * @param {Grid} grid
60127          */
60128         'render' : true,
60129             /**
60130              * @event select
60131              * Fires when a date is selected
60132              * @param {DatePicker} this
60133              * @param {Date} date The selected date
60134              */
60135         'select': true,
60136         /**
60137              * @event monthchange
60138              * Fires when the displayed month changes 
60139              * @param {DatePicker} this
60140              * @param {Date} date The selected month
60141              */
60142         'monthchange': true,
60143         /**
60144              * @event evententer
60145              * Fires when mouse over an event
60146              * @param {Calendar} this
60147              * @param {event} Event
60148              */
60149         'evententer': true,
60150         /**
60151              * @event eventleave
60152              * Fires when the mouse leaves an
60153              * @param {Calendar} this
60154              * @param {event}
60155              */
60156         'eventleave': true,
60157         /**
60158              * @event eventclick
60159              * Fires when the mouse click an
60160              * @param {Calendar} this
60161              * @param {event}
60162              */
60163         'eventclick': true,
60164         /**
60165              * @event eventrender
60166              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60167              * @param {Calendar} this
60168              * @param {data} data to be modified
60169              */
60170         'eventrender': true
60171         
60172     });
60173
60174     Roo.grid.Grid.superclass.constructor.call(this);
60175     this.on('render', function() {
60176         this.view.el.addClass('x-grid-cal'); 
60177         
60178         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60179
60180     },this);
60181     
60182     if (!Roo.grid.Calendar.style) {
60183         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60184             
60185             
60186             '.x-grid-cal .x-grid-col' :  {
60187                 height: 'auto !important',
60188                 'vertical-align': 'top'
60189             },
60190             '.x-grid-cal  .fc-event-hori' : {
60191                 height: '14px'
60192             }
60193              
60194             
60195         }, Roo.id());
60196     }
60197
60198     
60199     
60200 };
60201 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60202     /**
60203      * @cfg {Store} eventStore The store that loads events.
60204      */
60205     eventStore : 25,
60206
60207      
60208     activeDate : false,
60209     startDay : 0,
60210     autoWidth : true,
60211     monitorWindowResize : false,
60212
60213     
60214     resizeColumns : function() {
60215         var col = (this.view.el.getWidth() / 7) - 3;
60216         // loop through cols, and setWidth
60217         for(var i =0 ; i < 7 ; i++){
60218             this.cm.setColumnWidth(i, col);
60219         }
60220     },
60221      setDate :function(date) {
60222         
60223         Roo.log('setDate?');
60224         
60225         this.resizeColumns();
60226         var vd = this.activeDate;
60227         this.activeDate = date;
60228 //        if(vd && this.el){
60229 //            var t = date.getTime();
60230 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60231 //                Roo.log('using add remove');
60232 //                
60233 //                this.fireEvent('monthchange', this, date);
60234 //                
60235 //                this.cells.removeClass("fc-state-highlight");
60236 //                this.cells.each(function(c){
60237 //                   if(c.dateValue == t){
60238 //                       c.addClass("fc-state-highlight");
60239 //                       setTimeout(function(){
60240 //                            try{c.dom.firstChild.focus();}catch(e){}
60241 //                       }, 50);
60242 //                       return false;
60243 //                   }
60244 //                   return true;
60245 //                });
60246 //                return;
60247 //            }
60248 //        }
60249         
60250         var days = date.getDaysInMonth();
60251         
60252         var firstOfMonth = date.getFirstDateOfMonth();
60253         var startingPos = firstOfMonth.getDay()-this.startDay;
60254         
60255         if(startingPos < this.startDay){
60256             startingPos += 7;
60257         }
60258         
60259         var pm = date.add(Date.MONTH, -1);
60260         var prevStart = pm.getDaysInMonth()-startingPos;
60261 //        
60262         
60263         
60264         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60265         
60266         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60267         //this.cells.addClassOnOver('fc-state-hover');
60268         
60269         var cells = this.cells.elements;
60270         var textEls = this.textNodes;
60271         
60272         //Roo.each(cells, function(cell){
60273         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60274         //});
60275         
60276         days += startingPos;
60277
60278         // convert everything to numbers so it's fast
60279         var day = 86400000;
60280         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60281         //Roo.log(d);
60282         //Roo.log(pm);
60283         //Roo.log(prevStart);
60284         
60285         var today = new Date().clearTime().getTime();
60286         var sel = date.clearTime().getTime();
60287         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60288         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60289         var ddMatch = this.disabledDatesRE;
60290         var ddText = this.disabledDatesText;
60291         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60292         var ddaysText = this.disabledDaysText;
60293         var format = this.format;
60294         
60295         var setCellClass = function(cal, cell){
60296             
60297             //Roo.log('set Cell Class');
60298             cell.title = "";
60299             var t = d.getTime();
60300             
60301             //Roo.log(d);
60302             
60303             
60304             cell.dateValue = t;
60305             if(t == today){
60306                 cell.className += " fc-today";
60307                 cell.className += " fc-state-highlight";
60308                 cell.title = cal.todayText;
60309             }
60310             if(t == sel){
60311                 // disable highlight in other month..
60312                 cell.className += " fc-state-highlight";
60313                 
60314             }
60315             // disabling
60316             if(t < min) {
60317                 //cell.className = " fc-state-disabled";
60318                 cell.title = cal.minText;
60319                 return;
60320             }
60321             if(t > max) {
60322                 //cell.className = " fc-state-disabled";
60323                 cell.title = cal.maxText;
60324                 return;
60325             }
60326             if(ddays){
60327                 if(ddays.indexOf(d.getDay()) != -1){
60328                     // cell.title = ddaysText;
60329                    // cell.className = " fc-state-disabled";
60330                 }
60331             }
60332             if(ddMatch && format){
60333                 var fvalue = d.dateFormat(format);
60334                 if(ddMatch.test(fvalue)){
60335                     cell.title = ddText.replace("%0", fvalue);
60336                    cell.className = " fc-state-disabled";
60337                 }
60338             }
60339             
60340             if (!cell.initialClassName) {
60341                 cell.initialClassName = cell.dom.className;
60342             }
60343             
60344             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60345         };
60346
60347         var i = 0;
60348         
60349         for(; i < startingPos; i++) {
60350             cells[i].dayName =  (++prevStart);
60351             Roo.log(textEls[i]);
60352             d.setDate(d.getDate()+1);
60353             
60354             //cells[i].className = "fc-past fc-other-month";
60355             setCellClass(this, cells[i]);
60356         }
60357         
60358         var intDay = 0;
60359         
60360         for(; i < days; i++){
60361             intDay = i - startingPos + 1;
60362             cells[i].dayName =  (intDay);
60363             d.setDate(d.getDate()+1);
60364             
60365             cells[i].className = ''; // "x-date-active";
60366             setCellClass(this, cells[i]);
60367         }
60368         var extraDays = 0;
60369         
60370         for(; i < 42; i++) {
60371             //textEls[i].innerHTML = (++extraDays);
60372             
60373             d.setDate(d.getDate()+1);
60374             cells[i].dayName = (++extraDays);
60375             cells[i].className = "fc-future fc-other-month";
60376             setCellClass(this, cells[i]);
60377         }
60378         
60379         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60380         
60381         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60382         
60383         // this will cause all the cells to mis
60384         var rows= [];
60385         var i =0;
60386         for (var r = 0;r < 6;r++) {
60387             for (var c =0;c < 7;c++) {
60388                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60389             }    
60390         }
60391         
60392         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60393         for(i=0;i<cells.length;i++) {
60394             
60395             this.cells.elements[i].dayName = cells[i].dayName ;
60396             this.cells.elements[i].className = cells[i].className;
60397             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60398             this.cells.elements[i].title = cells[i].title ;
60399             this.cells.elements[i].dateValue = cells[i].dateValue ;
60400         }
60401         
60402         
60403         
60404         
60405         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60406         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60407         
60408         ////if(totalRows != 6){
60409             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60410            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60411        // }
60412         
60413         this.fireEvent('monthchange', this, date);
60414         
60415         
60416     },
60417  /**
60418      * Returns the grid's SelectionModel.
60419      * @return {SelectionModel}
60420      */
60421     getSelectionModel : function(){
60422         if(!this.selModel){
60423             this.selModel = new Roo.grid.CellSelectionModel();
60424         }
60425         return this.selModel;
60426     },
60427
60428     load: function() {
60429         this.eventStore.load()
60430         
60431         
60432         
60433     },
60434     
60435     findCell : function(dt) {
60436         dt = dt.clearTime().getTime();
60437         var ret = false;
60438         this.cells.each(function(c){
60439             //Roo.log("check " +c.dateValue + '?=' + dt);
60440             if(c.dateValue == dt){
60441                 ret = c;
60442                 return false;
60443             }
60444             return true;
60445         });
60446         
60447         return ret;
60448     },
60449     
60450     findCells : function(rec) {
60451         var s = rec.data.start_dt.clone().clearTime().getTime();
60452        // Roo.log(s);
60453         var e= rec.data.end_dt.clone().clearTime().getTime();
60454        // Roo.log(e);
60455         var ret = [];
60456         this.cells.each(function(c){
60457              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60458             
60459             if(c.dateValue > e){
60460                 return ;
60461             }
60462             if(c.dateValue < s){
60463                 return ;
60464             }
60465             ret.push(c);
60466         });
60467         
60468         return ret;    
60469     },
60470     
60471     findBestRow: function(cells)
60472     {
60473         var ret = 0;
60474         
60475         for (var i =0 ; i < cells.length;i++) {
60476             ret  = Math.max(cells[i].rows || 0,ret);
60477         }
60478         return ret;
60479         
60480     },
60481     
60482     
60483     addItem : function(rec)
60484     {
60485         // look for vertical location slot in
60486         var cells = this.findCells(rec);
60487         
60488         rec.row = this.findBestRow(cells);
60489         
60490         // work out the location.
60491         
60492         var crow = false;
60493         var rows = [];
60494         for(var i =0; i < cells.length; i++) {
60495             if (!crow) {
60496                 crow = {
60497                     start : cells[i],
60498                     end :  cells[i]
60499                 };
60500                 continue;
60501             }
60502             if (crow.start.getY() == cells[i].getY()) {
60503                 // on same row.
60504                 crow.end = cells[i];
60505                 continue;
60506             }
60507             // different row.
60508             rows.push(crow);
60509             crow = {
60510                 start: cells[i],
60511                 end : cells[i]
60512             };
60513             
60514         }
60515         
60516         rows.push(crow);
60517         rec.els = [];
60518         rec.rows = rows;
60519         rec.cells = cells;
60520         for (var i = 0; i < cells.length;i++) {
60521             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60522             
60523         }
60524         
60525         
60526     },
60527     
60528     clearEvents: function() {
60529         
60530         if (!this.eventStore.getCount()) {
60531             return;
60532         }
60533         // reset number of rows in cells.
60534         Roo.each(this.cells.elements, function(c){
60535             c.rows = 0;
60536         });
60537         
60538         this.eventStore.each(function(e) {
60539             this.clearEvent(e);
60540         },this);
60541         
60542     },
60543     
60544     clearEvent : function(ev)
60545     {
60546         if (ev.els) {
60547             Roo.each(ev.els, function(el) {
60548                 el.un('mouseenter' ,this.onEventEnter, this);
60549                 el.un('mouseleave' ,this.onEventLeave, this);
60550                 el.remove();
60551             },this);
60552             ev.els = [];
60553         }
60554     },
60555     
60556     
60557     renderEvent : function(ev,ctr) {
60558         if (!ctr) {
60559              ctr = this.view.el.select('.fc-event-container',true).first();
60560         }
60561         
60562          
60563         this.clearEvent(ev);
60564             //code
60565        
60566         
60567         
60568         ev.els = [];
60569         var cells = ev.cells;
60570         var rows = ev.rows;
60571         this.fireEvent('eventrender', this, ev);
60572         
60573         for(var i =0; i < rows.length; i++) {
60574             
60575             cls = '';
60576             if (i == 0) {
60577                 cls += ' fc-event-start';
60578             }
60579             if ((i+1) == rows.length) {
60580                 cls += ' fc-event-end';
60581             }
60582             
60583             //Roo.log(ev.data);
60584             // how many rows should it span..
60585             var cg = this.eventTmpl.append(ctr,Roo.apply({
60586                 fccls : cls
60587                 
60588             }, ev.data) , true);
60589             
60590             
60591             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60592             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60593             cg.on('click', this.onEventClick, this, ev);
60594             
60595             ev.els.push(cg);
60596             
60597             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60598             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60599             //Roo.log(cg);
60600              
60601             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60602             cg.setWidth(ebox.right - sbox.x -2);
60603         }
60604     },
60605     
60606     renderEvents: function()
60607     {   
60608         // first make sure there is enough space..
60609         
60610         if (!this.eventTmpl) {
60611             this.eventTmpl = new Roo.Template(
60612                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60613                     '<div class="fc-event-inner">' +
60614                         '<span class="fc-event-time">{time}</span>' +
60615                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60616                     '</div>' +
60617                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60618                 '</div>'
60619             );
60620                 
60621         }
60622                
60623         
60624         
60625         this.cells.each(function(c) {
60626             //Roo.log(c.select('.fc-day-content div',true).first());
60627             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60628         });
60629         
60630         var ctr = this.view.el.select('.fc-event-container',true).first();
60631         
60632         var cls;
60633         this.eventStore.each(function(ev){
60634             
60635             this.renderEvent(ev);
60636              
60637              
60638         }, this);
60639         this.view.layout();
60640         
60641     },
60642     
60643     onEventEnter: function (e, el,event,d) {
60644         this.fireEvent('evententer', this, el, event);
60645     },
60646     
60647     onEventLeave: function (e, el,event,d) {
60648         this.fireEvent('eventleave', this, el, event);
60649     },
60650     
60651     onEventClick: function (e, el,event,d) {
60652         this.fireEvent('eventclick', this, el, event);
60653     },
60654     
60655     onMonthChange: function () {
60656         this.store.load();
60657     },
60658     
60659     onLoad: function () {
60660         
60661         //Roo.log('calendar onload');
60662 //         
60663         if(this.eventStore.getCount() > 0){
60664             
60665            
60666             
60667             this.eventStore.each(function(d){
60668                 
60669                 
60670                 // FIXME..
60671                 var add =   d.data;
60672                 if (typeof(add.end_dt) == 'undefined')  {
60673                     Roo.log("Missing End time in calendar data: ");
60674                     Roo.log(d);
60675                     return;
60676                 }
60677                 if (typeof(add.start_dt) == 'undefined')  {
60678                     Roo.log("Missing Start time in calendar data: ");
60679                     Roo.log(d);
60680                     return;
60681                 }
60682                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60683                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60684                 add.id = add.id || d.id;
60685                 add.title = add.title || '??';
60686                 
60687                 this.addItem(d);
60688                 
60689              
60690             },this);
60691         }
60692         
60693         this.renderEvents();
60694     }
60695     
60696
60697 });
60698 /*
60699  grid : {
60700                 xtype: 'Grid',
60701                 xns: Roo.grid,
60702                 listeners : {
60703                     render : function ()
60704                     {
60705                         _this.grid = this;
60706                         
60707                         if (!this.view.el.hasClass('course-timesheet')) {
60708                             this.view.el.addClass('course-timesheet');
60709                         }
60710                         if (this.tsStyle) {
60711                             this.ds.load({});
60712                             return; 
60713                         }
60714                         Roo.log('width');
60715                         Roo.log(_this.grid.view.el.getWidth());
60716                         
60717                         
60718                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60719                             '.course-timesheet .x-grid-row' : {
60720                                 height: '80px'
60721                             },
60722                             '.x-grid-row td' : {
60723                                 'vertical-align' : 0
60724                             },
60725                             '.course-edit-link' : {
60726                                 'color' : 'blue',
60727                                 'text-overflow' : 'ellipsis',
60728                                 'overflow' : 'hidden',
60729                                 'white-space' : 'nowrap',
60730                                 'cursor' : 'pointer'
60731                             },
60732                             '.sub-link' : {
60733                                 'color' : 'green'
60734                             },
60735                             '.de-act-sup-link' : {
60736                                 'color' : 'purple',
60737                                 'text-decoration' : 'line-through'
60738                             },
60739                             '.de-act-link' : {
60740                                 'color' : 'red',
60741                                 'text-decoration' : 'line-through'
60742                             },
60743                             '.course-timesheet .course-highlight' : {
60744                                 'border-top-style': 'dashed !important',
60745                                 'border-bottom-bottom': 'dashed !important'
60746                             },
60747                             '.course-timesheet .course-item' : {
60748                                 'font-family'   : 'tahoma, arial, helvetica',
60749                                 'font-size'     : '11px',
60750                                 'overflow'      : 'hidden',
60751                                 'padding-left'  : '10px',
60752                                 'padding-right' : '10px',
60753                                 'padding-top' : '10px' 
60754                             }
60755                             
60756                         }, Roo.id());
60757                                 this.ds.load({});
60758                     }
60759                 },
60760                 autoWidth : true,
60761                 monitorWindowResize : false,
60762                 cellrenderer : function(v,x,r)
60763                 {
60764                     return v;
60765                 },
60766                 sm : {
60767                     xtype: 'CellSelectionModel',
60768                     xns: Roo.grid
60769                 },
60770                 dataSource : {
60771                     xtype: 'Store',
60772                     xns: Roo.data,
60773                     listeners : {
60774                         beforeload : function (_self, options)
60775                         {
60776                             options.params = options.params || {};
60777                             options.params._month = _this.monthField.getValue();
60778                             options.params.limit = 9999;
60779                             options.params['sort'] = 'when_dt';    
60780                             options.params['dir'] = 'ASC';    
60781                             this.proxy.loadResponse = this.loadResponse;
60782                             Roo.log("load?");
60783                             //this.addColumns();
60784                         },
60785                         load : function (_self, records, options)
60786                         {
60787                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60788                                 // if you click on the translation.. you can edit it...
60789                                 var el = Roo.get(this);
60790                                 var id = el.dom.getAttribute('data-id');
60791                                 var d = el.dom.getAttribute('data-date');
60792                                 var t = el.dom.getAttribute('data-time');
60793                                 //var id = this.child('span').dom.textContent;
60794                                 
60795                                 //Roo.log(this);
60796                                 Pman.Dialog.CourseCalendar.show({
60797                                     id : id,
60798                                     when_d : d,
60799                                     when_t : t,
60800                                     productitem_active : id ? 1 : 0
60801                                 }, function() {
60802                                     _this.grid.ds.load({});
60803                                 });
60804                            
60805                            });
60806                            
60807                            _this.panel.fireEvent('resize', [ '', '' ]);
60808                         }
60809                     },
60810                     loadResponse : function(o, success, response){
60811                             // this is overridden on before load..
60812                             
60813                             Roo.log("our code?");       
60814                             //Roo.log(success);
60815                             //Roo.log(response)
60816                             delete this.activeRequest;
60817                             if(!success){
60818                                 this.fireEvent("loadexception", this, o, response);
60819                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60820                                 return;
60821                             }
60822                             var result;
60823                             try {
60824                                 result = o.reader.read(response);
60825                             }catch(e){
60826                                 Roo.log("load exception?");
60827                                 this.fireEvent("loadexception", this, o, response, e);
60828                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60829                                 return;
60830                             }
60831                             Roo.log("ready...");        
60832                             // loop through result.records;
60833                             // and set this.tdate[date] = [] << array of records..
60834                             _this.tdata  = {};
60835                             Roo.each(result.records, function(r){
60836                                 //Roo.log(r.data);
60837                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60838                                     _this.tdata[r.data.when_dt.format('j')] = [];
60839                                 }
60840                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60841                             });
60842                             
60843                             //Roo.log(_this.tdata);
60844                             
60845                             result.records = [];
60846                             result.totalRecords = 6;
60847                     
60848                             // let's generate some duumy records for the rows.
60849                             //var st = _this.dateField.getValue();
60850                             
60851                             // work out monday..
60852                             //st = st.add(Date.DAY, -1 * st.format('w'));
60853                             
60854                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60855                             
60856                             var firstOfMonth = date.getFirstDayOfMonth();
60857                             var days = date.getDaysInMonth();
60858                             var d = 1;
60859                             var firstAdded = false;
60860                             for (var i = 0; i < result.totalRecords ; i++) {
60861                                 //var d= st.add(Date.DAY, i);
60862                                 var row = {};
60863                                 var added = 0;
60864                                 for(var w = 0 ; w < 7 ; w++){
60865                                     if(!firstAdded && firstOfMonth != w){
60866                                         continue;
60867                                     }
60868                                     if(d > days){
60869                                         continue;
60870                                     }
60871                                     firstAdded = true;
60872                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60873                                     row['weekday'+w] = String.format(
60874                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60875                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60876                                                     d,
60877                                                     date.format('Y-m-')+dd
60878                                                 );
60879                                     added++;
60880                                     if(typeof(_this.tdata[d]) != 'undefined'){
60881                                         Roo.each(_this.tdata[d], function(r){
60882                                             var is_sub = '';
60883                                             var deactive = '';
60884                                             var id = r.id;
60885                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60886                                             if(r.parent_id*1>0){
60887                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60888                                                 id = r.parent_id;
60889                                             }
60890                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60891                                                 deactive = 'de-act-link';
60892                                             }
60893                                             
60894                                             row['weekday'+w] += String.format(
60895                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60896                                                     id, //0
60897                                                     r.product_id_name, //1
60898                                                     r.when_dt.format('h:ia'), //2
60899                                                     is_sub, //3
60900                                                     deactive, //4
60901                                                     desc // 5
60902                                             );
60903                                         });
60904                                     }
60905                                     d++;
60906                                 }
60907                                 
60908                                 // only do this if something added..
60909                                 if(added > 0){ 
60910                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60911                                 }
60912                                 
60913                                 
60914                                 // push it twice. (second one with an hour..
60915                                 
60916                             }
60917                             //Roo.log(result);
60918                             this.fireEvent("load", this, o, o.request.arg);
60919                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60920                         },
60921                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60922                     proxy : {
60923                         xtype: 'HttpProxy',
60924                         xns: Roo.data,
60925                         method : 'GET',
60926                         url : baseURL + '/Roo/Shop_course.php'
60927                     },
60928                     reader : {
60929                         xtype: 'JsonReader',
60930                         xns: Roo.data,
60931                         id : 'id',
60932                         fields : [
60933                             {
60934                                 'name': 'id',
60935                                 'type': 'int'
60936                             },
60937                             {
60938                                 'name': 'when_dt',
60939                                 'type': 'string'
60940                             },
60941                             {
60942                                 'name': 'end_dt',
60943                                 'type': 'string'
60944                             },
60945                             {
60946                                 'name': 'parent_id',
60947                                 'type': 'int'
60948                             },
60949                             {
60950                                 'name': 'product_id',
60951                                 'type': 'int'
60952                             },
60953                             {
60954                                 'name': 'productitem_id',
60955                                 'type': 'int'
60956                             },
60957                             {
60958                                 'name': 'guid',
60959                                 'type': 'int'
60960                             }
60961                         ]
60962                     }
60963                 },
60964                 toolbar : {
60965                     xtype: 'Toolbar',
60966                     xns: Roo,
60967                     items : [
60968                         {
60969                             xtype: 'Button',
60970                             xns: Roo.Toolbar,
60971                             listeners : {
60972                                 click : function (_self, e)
60973                                 {
60974                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60975                                     sd.setMonth(sd.getMonth()-1);
60976                                     _this.monthField.setValue(sd.format('Y-m-d'));
60977                                     _this.grid.ds.load({});
60978                                 }
60979                             },
60980                             text : "Back"
60981                         },
60982                         {
60983                             xtype: 'Separator',
60984                             xns: Roo.Toolbar
60985                         },
60986                         {
60987                             xtype: 'MonthField',
60988                             xns: Roo.form,
60989                             listeners : {
60990                                 render : function (_self)
60991                                 {
60992                                     _this.monthField = _self;
60993                                    // _this.monthField.set  today
60994                                 },
60995                                 select : function (combo, date)
60996                                 {
60997                                     _this.grid.ds.load({});
60998                                 }
60999                             },
61000                             value : (function() { return new Date(); })()
61001                         },
61002                         {
61003                             xtype: 'Separator',
61004                             xns: Roo.Toolbar
61005                         },
61006                         {
61007                             xtype: 'TextItem',
61008                             xns: Roo.Toolbar,
61009                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
61010                         },
61011                         {
61012                             xtype: 'Fill',
61013                             xns: Roo.Toolbar
61014                         },
61015                         {
61016                             xtype: 'Button',
61017                             xns: Roo.Toolbar,
61018                             listeners : {
61019                                 click : function (_self, e)
61020                                 {
61021                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
61022                                     sd.setMonth(sd.getMonth()+1);
61023                                     _this.monthField.setValue(sd.format('Y-m-d'));
61024                                     _this.grid.ds.load({});
61025                                 }
61026                             },
61027                             text : "Next"
61028                         }
61029                     ]
61030                 },
61031                  
61032             }
61033         };
61034         
61035         *//*
61036  * Based on:
61037  * Ext JS Library 1.1.1
61038  * Copyright(c) 2006-2007, Ext JS, LLC.
61039  *
61040  * Originally Released Under LGPL - original licence link has changed is not relivant.
61041  *
61042  * Fork - LGPL
61043  * <script type="text/javascript">
61044  */
61045  
61046 /**
61047  * @class Roo.LoadMask
61048  * A simple utility class for generically masking elements while loading data.  If the element being masked has
61049  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
61050  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
61051  * element's UpdateManager load indicator and will be destroyed after the initial load.
61052  * @constructor
61053  * Create a new LoadMask
61054  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
61055  * @param {Object} config The config object
61056  */
61057 Roo.LoadMask = function(el, config){
61058     this.el = Roo.get(el);
61059     Roo.apply(this, config);
61060     if(this.store){
61061         this.store.on('beforeload', this.onBeforeLoad, this);
61062         this.store.on('load', this.onLoad, this);
61063         this.store.on('loadexception', this.onLoadException, this);
61064         this.removeMask = false;
61065     }else{
61066         var um = this.el.getUpdateManager();
61067         um.showLoadIndicator = false; // disable the default indicator
61068         um.on('beforeupdate', this.onBeforeLoad, this);
61069         um.on('update', this.onLoad, this);
61070         um.on('failure', this.onLoad, this);
61071         this.removeMask = true;
61072     }
61073 };
61074
61075 Roo.LoadMask.prototype = {
61076     /**
61077      * @cfg {Boolean} removeMask
61078      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61079      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61080      */
61081     /**
61082      * @cfg {String} msg
61083      * The text to display in a centered loading message box (defaults to 'Loading...')
61084      */
61085     msg : 'Loading...',
61086     /**
61087      * @cfg {String} msgCls
61088      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61089      */
61090     msgCls : 'x-mask-loading',
61091
61092     /**
61093      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61094      * @type Boolean
61095      */
61096     disabled: false,
61097
61098     /**
61099      * Disables the mask to prevent it from being displayed
61100      */
61101     disable : function(){
61102        this.disabled = true;
61103     },
61104
61105     /**
61106      * Enables the mask so that it can be displayed
61107      */
61108     enable : function(){
61109         this.disabled = false;
61110     },
61111     
61112     onLoadException : function()
61113     {
61114         Roo.log(arguments);
61115         
61116         if (typeof(arguments[3]) != 'undefined') {
61117             Roo.MessageBox.alert("Error loading",arguments[3]);
61118         } 
61119         /*
61120         try {
61121             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61122                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61123             }   
61124         } catch(e) {
61125             
61126         }
61127         */
61128     
61129         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61130     },
61131     // private
61132     onLoad : function()
61133     {
61134         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61135     },
61136
61137     // private
61138     onBeforeLoad : function(){
61139         if(!this.disabled){
61140             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61141         }
61142     },
61143
61144     // private
61145     destroy : function(){
61146         if(this.store){
61147             this.store.un('beforeload', this.onBeforeLoad, this);
61148             this.store.un('load', this.onLoad, this);
61149             this.store.un('loadexception', this.onLoadException, this);
61150         }else{
61151             var um = this.el.getUpdateManager();
61152             um.un('beforeupdate', this.onBeforeLoad, this);
61153             um.un('update', this.onLoad, this);
61154             um.un('failure', this.onLoad, this);
61155         }
61156     }
61157 };/*
61158  * Based on:
61159  * Ext JS Library 1.1.1
61160  * Copyright(c) 2006-2007, Ext JS, LLC.
61161  *
61162  * Originally Released Under LGPL - original licence link has changed is not relivant.
61163  *
61164  * Fork - LGPL
61165  * <script type="text/javascript">
61166  */
61167
61168
61169 /**
61170  * @class Roo.XTemplate
61171  * @extends Roo.Template
61172  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61173 <pre><code>
61174 var t = new Roo.XTemplate(
61175         '&lt;select name="{name}"&gt;',
61176                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61177         '&lt;/select&gt;'
61178 );
61179  
61180 // then append, applying the master template values
61181  </code></pre>
61182  *
61183  * Supported features:
61184  *
61185  *  Tags:
61186
61187 <pre><code>
61188       {a_variable} - output encoded.
61189       {a_variable.format:("Y-m-d")} - call a method on the variable
61190       {a_variable:raw} - unencoded output
61191       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61192       {a_variable:this.method_on_template(...)} - call a method on the template object.
61193  
61194 </code></pre>
61195  *  The tpl tag:
61196 <pre><code>
61197         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61198         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61199         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61200         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61201   
61202         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61203         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61204 </code></pre>
61205  *      
61206  */
61207 Roo.XTemplate = function()
61208 {
61209     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61210     if (this.html) {
61211         this.compile();
61212     }
61213 };
61214
61215
61216 Roo.extend(Roo.XTemplate, Roo.Template, {
61217
61218     /**
61219      * The various sub templates
61220      */
61221     tpls : false,
61222     /**
61223      *
61224      * basic tag replacing syntax
61225      * WORD:WORD()
61226      *
61227      * // you can fake an object call by doing this
61228      *  x.t:(test,tesT) 
61229      * 
61230      */
61231     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61232
61233     /**
61234      * compile the template
61235      *
61236      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61237      *
61238      */
61239     compile: function()
61240     {
61241         var s = this.html;
61242      
61243         s = ['<tpl>', s, '</tpl>'].join('');
61244     
61245         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61246             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61247             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61248             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61249             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61250             m,
61251             id     = 0,
61252             tpls   = [];
61253     
61254         while(true == !!(m = s.match(re))){
61255             var forMatch   = m[0].match(nameRe),
61256                 ifMatch   = m[0].match(ifRe),
61257                 execMatch   = m[0].match(execRe),
61258                 namedMatch   = m[0].match(namedRe),
61259                 
61260                 exp  = null, 
61261                 fn   = null,
61262                 exec = null,
61263                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61264                 
61265             if (ifMatch) {
61266                 // if - puts fn into test..
61267                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61268                 if(exp){
61269                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61270                 }
61271             }
61272             
61273             if (execMatch) {
61274                 // exec - calls a function... returns empty if true is  returned.
61275                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61276                 if(exp){
61277                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61278                 }
61279             }
61280             
61281             
61282             if (name) {
61283                 // for = 
61284                 switch(name){
61285                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61286                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61287                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61288                 }
61289             }
61290             var uid = namedMatch ? namedMatch[1] : id;
61291             
61292             
61293             tpls.push({
61294                 id:     namedMatch ? namedMatch[1] : id,
61295                 target: name,
61296                 exec:   exec,
61297                 test:   fn,
61298                 body:   m[1] || ''
61299             });
61300             if (namedMatch) {
61301                 s = s.replace(m[0], '');
61302             } else { 
61303                 s = s.replace(m[0], '{xtpl'+ id + '}');
61304             }
61305             ++id;
61306         }
61307         this.tpls = [];
61308         for(var i = tpls.length-1; i >= 0; --i){
61309             this.compileTpl(tpls[i]);
61310             this.tpls[tpls[i].id] = tpls[i];
61311         }
61312         this.master = tpls[tpls.length-1];
61313         return this;
61314     },
61315     /**
61316      * same as applyTemplate, except it's done to one of the subTemplates
61317      * when using named templates, you can do:
61318      *
61319      * var str = pl.applySubTemplate('your-name', values);
61320      *
61321      * 
61322      * @param {Number} id of the template
61323      * @param {Object} values to apply to template
61324      * @param {Object} parent (normaly the instance of this object)
61325      */
61326     applySubTemplate : function(id, values, parent)
61327     {
61328         
61329         
61330         var t = this.tpls[id];
61331         
61332         
61333         try { 
61334             if(t.test && !t.test.call(this, values, parent)){
61335                 return '';
61336             }
61337         } catch(e) {
61338             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61339             Roo.log(e.toString());
61340             Roo.log(t.test);
61341             return ''
61342         }
61343         try { 
61344             
61345             if(t.exec && t.exec.call(this, values, parent)){
61346                 return '';
61347             }
61348         } catch(e) {
61349             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61350             Roo.log(e.toString());
61351             Roo.log(t.exec);
61352             return ''
61353         }
61354         try {
61355             var vs = t.target ? t.target.call(this, values, parent) : values;
61356             parent = t.target ? values : parent;
61357             if(t.target && vs instanceof Array){
61358                 var buf = [];
61359                 for(var i = 0, len = vs.length; i < len; i++){
61360                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61361                 }
61362                 return buf.join('');
61363             }
61364             return t.compiled.call(this, vs, parent);
61365         } catch (e) {
61366             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61367             Roo.log(e.toString());
61368             Roo.log(t.compiled);
61369             return '';
61370         }
61371     },
61372
61373     compileTpl : function(tpl)
61374     {
61375         var fm = Roo.util.Format;
61376         var useF = this.disableFormats !== true;
61377         var sep = Roo.isGecko ? "+" : ",";
61378         var undef = function(str) {
61379             Roo.log("Property not found :"  + str);
61380             return '';
61381         };
61382         
61383         var fn = function(m, name, format, args)
61384         {
61385             //Roo.log(arguments);
61386             args = args ? args.replace(/\\'/g,"'") : args;
61387             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61388             if (typeof(format) == 'undefined') {
61389                 format= 'htmlEncode';
61390             }
61391             if (format == 'raw' ) {
61392                 format = false;
61393             }
61394             
61395             if(name.substr(0, 4) == 'xtpl'){
61396                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61397             }
61398             
61399             // build an array of options to determine if value is undefined..
61400             
61401             // basically get 'xxxx.yyyy' then do
61402             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61403             //    (function () { Roo.log("Property not found"); return ''; })() :
61404             //    ......
61405             
61406             var udef_ar = [];
61407             var lookfor = '';
61408             Roo.each(name.split('.'), function(st) {
61409                 lookfor += (lookfor.length ? '.': '') + st;
61410                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61411             });
61412             
61413             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61414             
61415             
61416             if(format && useF){
61417                 
61418                 args = args ? ',' + args : "";
61419                  
61420                 if(format.substr(0, 5) != "this."){
61421                     format = "fm." + format + '(';
61422                 }else{
61423                     format = 'this.call("'+ format.substr(5) + '", ';
61424                     args = ", values";
61425                 }
61426                 
61427                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61428             }
61429              
61430             if (args.length) {
61431                 // called with xxyx.yuu:(test,test)
61432                 // change to ()
61433                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61434             }
61435             // raw.. - :raw modifier..
61436             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61437             
61438         };
61439         var body;
61440         // branched to use + in gecko and [].join() in others
61441         if(Roo.isGecko){
61442             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61443                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61444                     "';};};";
61445         }else{
61446             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61447             body.push(tpl.body.replace(/(\r\n|\n)/g,
61448                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61449             body.push("'].join('');};};");
61450             body = body.join('');
61451         }
61452         
61453         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61454        
61455         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61456         eval(body);
61457         
61458         return this;
61459     },
61460
61461     applyTemplate : function(values){
61462         return this.master.compiled.call(this, values, {});
61463         //var s = this.subs;
61464     },
61465
61466     apply : function(){
61467         return this.applyTemplate.apply(this, arguments);
61468     }
61469
61470  });
61471
61472 Roo.XTemplate.from = function(el){
61473     el = Roo.getDom(el);
61474     return new Roo.XTemplate(el.value || el.innerHTML);
61475 };