sync
[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                 var args = Array.prototype.slice.call(arguments, 0);                
6078                 for(var i = 0; i < len; i++){
6079                     var l = ls[i];
6080                     if(l.fireFn.apply(l.scope||this.obj||window, args) === false){
6081                         this.firing = false;
6082                         return false;
6083                     }
6084                 }
6085                 this.firing = false;
6086             }
6087             return true;
6088         }
6089     };
6090 })();/*
6091  * RooJS Library 
6092  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6093  *
6094  * Licence LGPL 
6095  *
6096  */
6097  
6098 /**
6099  * @class Roo.Document
6100  * @extends Roo.util.Observable
6101  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6102  * 
6103  * @param {Object} config the methods and properties of the 'base' class for the application.
6104  * 
6105  *  Generic Page handler - implement this to start your app..
6106  * 
6107  * eg.
6108  *  MyProject = new Roo.Document({
6109         events : {
6110             'load' : true // your events..
6111         },
6112         listeners : {
6113             'ready' : function() {
6114                 // fired on Roo.onReady()
6115             }
6116         }
6117  * 
6118  */
6119 Roo.Document = function(cfg) {
6120      
6121     this.addEvents({ 
6122         'ready' : true
6123     });
6124     Roo.util.Observable.call(this,cfg);
6125     
6126     var _this = this;
6127     
6128     Roo.onReady(function() {
6129         _this.fireEvent('ready');
6130     },null,false);
6131     
6132     
6133 }
6134
6135 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6136  * Based on:
6137  * Ext JS Library 1.1.1
6138  * Copyright(c) 2006-2007, Ext JS, LLC.
6139  *
6140  * Originally Released Under LGPL - original licence link has changed is not relivant.
6141  *
6142  * Fork - LGPL
6143  * <script type="text/javascript">
6144  */
6145
6146 /**
6147  * @class Roo.EventManager
6148  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6149  * several useful events directly.
6150  * See {@link Roo.EventObject} for more details on normalized event objects.
6151  * @singleton
6152  */
6153 Roo.EventManager = function(){
6154     var docReadyEvent, docReadyProcId, docReadyState = false;
6155     var resizeEvent, resizeTask, textEvent, textSize;
6156     var E = Roo.lib.Event;
6157     var D = Roo.lib.Dom;
6158
6159     
6160     
6161
6162     var fireDocReady = function(){
6163         if(!docReadyState){
6164             docReadyState = true;
6165             Roo.isReady = true;
6166             if(docReadyProcId){
6167                 clearInterval(docReadyProcId);
6168             }
6169             if(Roo.isGecko || Roo.isOpera) {
6170                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6171             }
6172             if(Roo.isIE){
6173                 var defer = document.getElementById("ie-deferred-loader");
6174                 if(defer){
6175                     defer.onreadystatechange = null;
6176                     defer.parentNode.removeChild(defer);
6177                 }
6178             }
6179             if(docReadyEvent){
6180                 docReadyEvent.fire();
6181                 docReadyEvent.clearListeners();
6182             }
6183         }
6184     };
6185     
6186     var initDocReady = function(){
6187         docReadyEvent = new Roo.util.Event();
6188         if(Roo.isGecko || Roo.isOpera) {
6189             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6190         }else if(Roo.isIE){
6191             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6192             var defer = document.getElementById("ie-deferred-loader");
6193             defer.onreadystatechange = function(){
6194                 if(this.readyState == "complete"){
6195                     fireDocReady();
6196                 }
6197             };
6198         }else if(Roo.isSafari){ 
6199             docReadyProcId = setInterval(function(){
6200                 var rs = document.readyState;
6201                 if(rs == "complete") {
6202                     fireDocReady();     
6203                  }
6204             }, 10);
6205         }
6206         // no matter what, make sure it fires on load
6207         E.on(window, "load", fireDocReady);
6208     };
6209
6210     var createBuffered = function(h, o){
6211         var task = new Roo.util.DelayedTask(h);
6212         return function(e){
6213             // create new event object impl so new events don't wipe out properties
6214             e = new Roo.EventObjectImpl(e);
6215             task.delay(o.buffer, h, null, [e]);
6216         };
6217     };
6218
6219     var createSingle = function(h, el, ename, fn){
6220         return function(e){
6221             Roo.EventManager.removeListener(el, ename, fn);
6222             h(e);
6223         };
6224     };
6225
6226     var createDelayed = function(h, o){
6227         return function(e){
6228             // create new event object impl so new events don't wipe out properties
6229             e = new Roo.EventObjectImpl(e);
6230             setTimeout(function(){
6231                 h(e);
6232             }, o.delay || 10);
6233         };
6234     };
6235     var transitionEndVal = false;
6236     
6237     var transitionEnd = function()
6238     {
6239         if (transitionEndVal) {
6240             return transitionEndVal;
6241         }
6242         var el = document.createElement('div');
6243
6244         var transEndEventNames = {
6245             WebkitTransition : 'webkitTransitionEnd',
6246             MozTransition    : 'transitionend',
6247             OTransition      : 'oTransitionEnd otransitionend',
6248             transition       : 'transitionend'
6249         };
6250     
6251         for (var name in transEndEventNames) {
6252             if (el.style[name] !== undefined) {
6253                 transitionEndVal = transEndEventNames[name];
6254                 return  transitionEndVal ;
6255             }
6256         }
6257     }
6258     
6259
6260     var listen = function(element, ename, opt, fn, scope){
6261         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6262         fn = fn || o.fn; scope = scope || o.scope;
6263         var el = Roo.getDom(element);
6264         
6265         
6266         if(!el){
6267             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6268         }
6269         
6270         if (ename == 'transitionend') {
6271             ename = transitionEnd();
6272         }
6273         var h = function(e){
6274             e = Roo.EventObject.setEvent(e);
6275             var t;
6276             if(o.delegate){
6277                 t = e.getTarget(o.delegate, el);
6278                 if(!t){
6279                     return;
6280                 }
6281             }else{
6282                 t = e.target;
6283             }
6284             if(o.stopEvent === true){
6285                 e.stopEvent();
6286             }
6287             if(o.preventDefault === true){
6288                e.preventDefault();
6289             }
6290             if(o.stopPropagation === true){
6291                 e.stopPropagation();
6292             }
6293
6294             if(o.normalized === false){
6295                 e = e.browserEvent;
6296             }
6297
6298             fn.call(scope || el, e, t, o);
6299         };
6300         if(o.delay){
6301             h = createDelayed(h, o);
6302         }
6303         if(o.single){
6304             h = createSingle(h, el, ename, fn);
6305         }
6306         if(o.buffer){
6307             h = createBuffered(h, o);
6308         }
6309         
6310         fn._handlers = fn._handlers || [];
6311         
6312         
6313         fn._handlers.push([Roo.id(el), ename, h]);
6314         
6315         
6316          
6317         E.on(el, ename, h);
6318         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6319             el.addEventListener("DOMMouseScroll", h, false);
6320             E.on(window, 'unload', function(){
6321                 el.removeEventListener("DOMMouseScroll", h, false);
6322             });
6323         }
6324         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6325             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6326         }
6327         return h;
6328     };
6329
6330     var stopListening = function(el, ename, fn){
6331         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6332         if(hds){
6333             for(var i = 0, len = hds.length; i < len; i++){
6334                 var h = hds[i];
6335                 if(h[0] == id && h[1] == ename){
6336                     hd = h[2];
6337                     hds.splice(i, 1);
6338                     break;
6339                 }
6340             }
6341         }
6342         E.un(el, ename, hd);
6343         el = Roo.getDom(el);
6344         if(ename == "mousewheel" && el.addEventListener){
6345             el.removeEventListener("DOMMouseScroll", hd, false);
6346         }
6347         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6348             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6349         }
6350     };
6351
6352     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6353     
6354     var pub = {
6355         
6356         
6357         /** 
6358          * Fix for doc tools
6359          * @scope Roo.EventManager
6360          */
6361         
6362         
6363         /** 
6364          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6365          * object with a Roo.EventObject
6366          * @param {Function} fn        The method the event invokes
6367          * @param {Object}   scope    An object that becomes the scope of the handler
6368          * @param {boolean}  override If true, the obj passed in becomes
6369          *                             the execution scope of the listener
6370          * @return {Function} The wrapped function
6371          * @deprecated
6372          */
6373         wrap : function(fn, scope, override){
6374             return function(e){
6375                 Roo.EventObject.setEvent(e);
6376                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6377             };
6378         },
6379         
6380         /**
6381      * Appends an event handler to an element (shorthand for addListener)
6382      * @param {String/HTMLElement}   element        The html element or id to assign the
6383      * @param {String}   eventName The type of event to listen for
6384      * @param {Function} handler The method the event invokes
6385      * @param {Object}   scope (optional) The scope in which to execute the handler
6386      * function. The handler function's "this" context.
6387      * @param {Object}   options (optional) An object containing handler configuration
6388      * properties. This may contain any of the following properties:<ul>
6389      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6390      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6391      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6392      * <li>preventDefault {Boolean} True to prevent the default action</li>
6393      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6394      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6395      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6396      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6397      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6398      * by the specified number of milliseconds. If the event fires again within that time, the original
6399      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6400      * </ul><br>
6401      * <p>
6402      * <b>Combining Options</b><br>
6403      * Using the options argument, it is possible to combine different types of listeners:<br>
6404      * <br>
6405      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6406      * Code:<pre><code>
6407 el.on('click', this.onClick, this, {
6408     single: true,
6409     delay: 100,
6410     stopEvent : true,
6411     forumId: 4
6412 });</code></pre>
6413      * <p>
6414      * <b>Attaching multiple handlers in 1 call</b><br>
6415       * The method also allows for a single argument to be passed which is a config object containing properties
6416      * which specify multiple handlers.
6417      * <p>
6418      * Code:<pre><code>
6419 el.on({
6420     'click' : {
6421         fn: this.onClick
6422         scope: this,
6423         delay: 100
6424     },
6425     'mouseover' : {
6426         fn: this.onMouseOver
6427         scope: this
6428     },
6429     'mouseout' : {
6430         fn: this.onMouseOut
6431         scope: this
6432     }
6433 });</code></pre>
6434      * <p>
6435      * Or a shorthand syntax:<br>
6436      * Code:<pre><code>
6437 el.on({
6438     'click' : this.onClick,
6439     'mouseover' : this.onMouseOver,
6440     'mouseout' : this.onMouseOut
6441     scope: this
6442 });</code></pre>
6443      */
6444         addListener : function(element, eventName, fn, scope, options){
6445             if(typeof eventName == "object"){
6446                 var o = eventName;
6447                 for(var e in o){
6448                     if(propRe.test(e)){
6449                         continue;
6450                     }
6451                     if(typeof o[e] == "function"){
6452                         // shared options
6453                         listen(element, e, o, o[e], o.scope);
6454                     }else{
6455                         // individual options
6456                         listen(element, e, o[e]);
6457                     }
6458                 }
6459                 return;
6460             }
6461             return listen(element, eventName, options, fn, scope);
6462         },
6463         
6464         /**
6465          * Removes an event handler
6466          *
6467          * @param {String/HTMLElement}   element        The id or html element to remove the 
6468          *                             event from
6469          * @param {String}   eventName     The type of event
6470          * @param {Function} fn
6471          * @return {Boolean} True if a listener was actually removed
6472          */
6473         removeListener : function(element, eventName, fn){
6474             return stopListening(element, eventName, fn);
6475         },
6476         
6477         /**
6478          * Fires when the document is ready (before onload and before images are loaded). Can be 
6479          * accessed shorthanded Roo.onReady().
6480          * @param {Function} fn        The method the event invokes
6481          * @param {Object}   scope    An  object that becomes the scope of the handler
6482          * @param {boolean}  options
6483          */
6484         onDocumentReady : function(fn, scope, options){
6485             if(docReadyState){ // if it already fired
6486                 docReadyEvent.addListener(fn, scope, options);
6487                 docReadyEvent.fire();
6488                 docReadyEvent.clearListeners();
6489                 return;
6490             }
6491             if(!docReadyEvent){
6492                 initDocReady();
6493             }
6494             docReadyEvent.addListener(fn, scope, options);
6495         },
6496         
6497         /**
6498          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6499          * @param {Function} fn        The method the event invokes
6500          * @param {Object}   scope    An object that becomes the scope of the handler
6501          * @param {boolean}  options
6502          */
6503         onWindowResize : function(fn, scope, options){
6504             if(!resizeEvent){
6505                 resizeEvent = new Roo.util.Event();
6506                 resizeTask = new Roo.util.DelayedTask(function(){
6507                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6508                 });
6509                 E.on(window, "resize", function(){
6510                     if(Roo.isIE){
6511                         resizeTask.delay(50);
6512                     }else{
6513                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6514                     }
6515                 });
6516             }
6517             resizeEvent.addListener(fn, scope, options);
6518         },
6519
6520         /**
6521          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6522          * @param {Function} fn        The method the event invokes
6523          * @param {Object}   scope    An object that becomes the scope of the handler
6524          * @param {boolean}  options
6525          */
6526         onTextResize : function(fn, scope, options){
6527             if(!textEvent){
6528                 textEvent = new Roo.util.Event();
6529                 var textEl = new Roo.Element(document.createElement('div'));
6530                 textEl.dom.className = 'x-text-resize';
6531                 textEl.dom.innerHTML = 'X';
6532                 textEl.appendTo(document.body);
6533                 textSize = textEl.dom.offsetHeight;
6534                 setInterval(function(){
6535                     if(textEl.dom.offsetHeight != textSize){
6536                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6537                     }
6538                 }, this.textResizeInterval);
6539             }
6540             textEvent.addListener(fn, scope, options);
6541         },
6542
6543         /**
6544          * Removes the passed window resize listener.
6545          * @param {Function} fn        The method the event invokes
6546          * @param {Object}   scope    The scope of handler
6547          */
6548         removeResizeListener : function(fn, scope){
6549             if(resizeEvent){
6550                 resizeEvent.removeListener(fn, scope);
6551             }
6552         },
6553
6554         // private
6555         fireResize : function(){
6556             if(resizeEvent){
6557                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6558             }   
6559         },
6560         /**
6561          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6562          */
6563         ieDeferSrc : false,
6564         /**
6565          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6566          */
6567         textResizeInterval : 50
6568     };
6569     
6570     /**
6571      * Fix for doc tools
6572      * @scopeAlias pub=Roo.EventManager
6573      */
6574     
6575      /**
6576      * Appends an event handler to an element (shorthand for addListener)
6577      * @param {String/HTMLElement}   element        The html element or id to assign the
6578      * @param {String}   eventName The type of event to listen for
6579      * @param {Function} handler The method the event invokes
6580      * @param {Object}   scope (optional) The scope in which to execute the handler
6581      * function. The handler function's "this" context.
6582      * @param {Object}   options (optional) An object containing handler configuration
6583      * properties. This may contain any of the following properties:<ul>
6584      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6585      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6586      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6587      * <li>preventDefault {Boolean} True to prevent the default action</li>
6588      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6589      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6590      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6591      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6592      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6593      * by the specified number of milliseconds. If the event fires again within that time, the original
6594      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6595      * </ul><br>
6596      * <p>
6597      * <b>Combining Options</b><br>
6598      * Using the options argument, it is possible to combine different types of listeners:<br>
6599      * <br>
6600      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6601      * Code:<pre><code>
6602 el.on('click', this.onClick, this, {
6603     single: true,
6604     delay: 100,
6605     stopEvent : true,
6606     forumId: 4
6607 });</code></pre>
6608      * <p>
6609      * <b>Attaching multiple handlers in 1 call</b><br>
6610       * The method also allows for a single argument to be passed which is a config object containing properties
6611      * which specify multiple handlers.
6612      * <p>
6613      * Code:<pre><code>
6614 el.on({
6615     'click' : {
6616         fn: this.onClick
6617         scope: this,
6618         delay: 100
6619     },
6620     'mouseover' : {
6621         fn: this.onMouseOver
6622         scope: this
6623     },
6624     'mouseout' : {
6625         fn: this.onMouseOut
6626         scope: this
6627     }
6628 });</code></pre>
6629      * <p>
6630      * Or a shorthand syntax:<br>
6631      * Code:<pre><code>
6632 el.on({
6633     'click' : this.onClick,
6634     'mouseover' : this.onMouseOver,
6635     'mouseout' : this.onMouseOut
6636     scope: this
6637 });</code></pre>
6638      */
6639     pub.on = pub.addListener;
6640     pub.un = pub.removeListener;
6641
6642     pub.stoppedMouseDownEvent = new Roo.util.Event();
6643     return pub;
6644 }();
6645 /**
6646   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6647   * @param {Function} fn        The method the event invokes
6648   * @param {Object}   scope    An  object that becomes the scope of the handler
6649   * @param {boolean}  override If true, the obj passed in becomes
6650   *                             the execution scope of the listener
6651   * @member Roo
6652   * @method onReady
6653  */
6654 Roo.onReady = Roo.EventManager.onDocumentReady;
6655
6656 Roo.onReady(function(){
6657     var bd = Roo.get(document.body);
6658     if(!bd){ return; }
6659
6660     var cls = [
6661             Roo.isIE ? "roo-ie"
6662             : Roo.isIE11 ? "roo-ie11"
6663             : Roo.isEdge ? "roo-edge"
6664             : Roo.isGecko ? "roo-gecko"
6665             : Roo.isOpera ? "roo-opera"
6666             : Roo.isSafari ? "roo-safari" : ""];
6667
6668     if(Roo.isMac){
6669         cls.push("roo-mac");
6670     }
6671     if(Roo.isLinux){
6672         cls.push("roo-linux");
6673     }
6674     if(Roo.isIOS){
6675         cls.push("roo-ios");
6676     }
6677     if(Roo.isTouch){
6678         cls.push("roo-touch");
6679     }
6680     if(Roo.isBorderBox){
6681         cls.push('roo-border-box');
6682     }
6683     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6684         var p = bd.dom.parentNode;
6685         if(p){
6686             p.className += ' roo-strict';
6687         }
6688     }
6689     bd.addClass(cls.join(' '));
6690 });
6691
6692 /**
6693  * @class Roo.EventObject
6694  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6695  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6696  * Example:
6697  * <pre><code>
6698  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6699     e.preventDefault();
6700     var target = e.getTarget();
6701     ...
6702  }
6703  var myDiv = Roo.get("myDiv");
6704  myDiv.on("click", handleClick);
6705  //or
6706  Roo.EventManager.on("myDiv", 'click', handleClick);
6707  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6708  </code></pre>
6709  * @singleton
6710  */
6711 Roo.EventObject = function(){
6712     
6713     var E = Roo.lib.Event;
6714     
6715     // safari keypress events for special keys return bad keycodes
6716     var safariKeys = {
6717         63234 : 37, // left
6718         63235 : 39, // right
6719         63232 : 38, // up
6720         63233 : 40, // down
6721         63276 : 33, // page up
6722         63277 : 34, // page down
6723         63272 : 46, // delete
6724         63273 : 36, // home
6725         63275 : 35  // end
6726     };
6727
6728     // normalize button clicks
6729     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6730                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6731
6732     Roo.EventObjectImpl = function(e){
6733         if(e){
6734             this.setEvent(e.browserEvent || e);
6735         }
6736     };
6737     Roo.EventObjectImpl.prototype = {
6738         /**
6739          * Used to fix doc tools.
6740          * @scope Roo.EventObject.prototype
6741          */
6742             
6743
6744         
6745         
6746         /** The normal browser event */
6747         browserEvent : null,
6748         /** The button pressed in a mouse event */
6749         button : -1,
6750         /** True if the shift key was down during the event */
6751         shiftKey : false,
6752         /** True if the control key was down during the event */
6753         ctrlKey : false,
6754         /** True if the alt key was down during the event */
6755         altKey : false,
6756
6757         /** Key constant 
6758         * @type Number */
6759         BACKSPACE : 8,
6760         /** Key constant 
6761         * @type Number */
6762         TAB : 9,
6763         /** Key constant 
6764         * @type Number */
6765         RETURN : 13,
6766         /** Key constant 
6767         * @type Number */
6768         ENTER : 13,
6769         /** Key constant 
6770         * @type Number */
6771         SHIFT : 16,
6772         /** Key constant 
6773         * @type Number */
6774         CONTROL : 17,
6775         /** Key constant 
6776         * @type Number */
6777         ESC : 27,
6778         /** Key constant 
6779         * @type Number */
6780         SPACE : 32,
6781         /** Key constant 
6782         * @type Number */
6783         PAGEUP : 33,
6784         /** Key constant 
6785         * @type Number */
6786         PAGEDOWN : 34,
6787         /** Key constant 
6788         * @type Number */
6789         END : 35,
6790         /** Key constant 
6791         * @type Number */
6792         HOME : 36,
6793         /** Key constant 
6794         * @type Number */
6795         LEFT : 37,
6796         /** Key constant 
6797         * @type Number */
6798         UP : 38,
6799         /** Key constant 
6800         * @type Number */
6801         RIGHT : 39,
6802         /** Key constant 
6803         * @type Number */
6804         DOWN : 40,
6805         /** Key constant 
6806         * @type Number */
6807         DELETE : 46,
6808         /** Key constant 
6809         * @type Number */
6810         F5 : 116,
6811
6812            /** @private */
6813         setEvent : function(e){
6814             if(e == this || (e && e.browserEvent)){ // already wrapped
6815                 return e;
6816             }
6817             this.browserEvent = e;
6818             if(e){
6819                 // normalize buttons
6820                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6821                 if(e.type == 'click' && this.button == -1){
6822                     this.button = 0;
6823                 }
6824                 this.type = e.type;
6825                 this.shiftKey = e.shiftKey;
6826                 // mac metaKey behaves like ctrlKey
6827                 this.ctrlKey = e.ctrlKey || e.metaKey;
6828                 this.altKey = e.altKey;
6829                 // in getKey these will be normalized for the mac
6830                 this.keyCode = e.keyCode;
6831                 // keyup warnings on firefox.
6832                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6833                 // cache the target for the delayed and or buffered events
6834                 this.target = E.getTarget(e);
6835                 // same for XY
6836                 this.xy = E.getXY(e);
6837             }else{
6838                 this.button = -1;
6839                 this.shiftKey = false;
6840                 this.ctrlKey = false;
6841                 this.altKey = false;
6842                 this.keyCode = 0;
6843                 this.charCode =0;
6844                 this.target = null;
6845                 this.xy = [0, 0];
6846             }
6847             return this;
6848         },
6849
6850         /**
6851          * Stop the event (preventDefault and stopPropagation)
6852          */
6853         stopEvent : function(){
6854             if(this.browserEvent){
6855                 if(this.browserEvent.type == 'mousedown'){
6856                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6857                 }
6858                 E.stopEvent(this.browserEvent);
6859             }
6860         },
6861
6862         /**
6863          * Prevents the browsers default handling of the event.
6864          */
6865         preventDefault : function(){
6866             if(this.browserEvent){
6867                 E.preventDefault(this.browserEvent);
6868             }
6869         },
6870
6871         /** @private */
6872         isNavKeyPress : function(){
6873             var k = this.keyCode;
6874             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6875             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6876         },
6877
6878         isSpecialKey : function(){
6879             var k = this.keyCode;
6880             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6881             (k == 16) || (k == 17) ||
6882             (k >= 18 && k <= 20) ||
6883             (k >= 33 && k <= 35) ||
6884             (k >= 36 && k <= 39) ||
6885             (k >= 44 && k <= 45);
6886         },
6887         /**
6888          * Cancels bubbling of the event.
6889          */
6890         stopPropagation : function(){
6891             if(this.browserEvent){
6892                 if(this.type == 'mousedown'){
6893                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6894                 }
6895                 E.stopPropagation(this.browserEvent);
6896             }
6897         },
6898
6899         /**
6900          * Gets the key code for the event.
6901          * @return {Number}
6902          */
6903         getCharCode : function(){
6904             return this.charCode || this.keyCode;
6905         },
6906
6907         /**
6908          * Returns a normalized keyCode for the event.
6909          * @return {Number} The key code
6910          */
6911         getKey : function(){
6912             var k = this.keyCode || this.charCode;
6913             return Roo.isSafari ? (safariKeys[k] || k) : k;
6914         },
6915
6916         /**
6917          * Gets the x coordinate of the event.
6918          * @return {Number}
6919          */
6920         getPageX : function(){
6921             return this.xy[0];
6922         },
6923
6924         /**
6925          * Gets the y coordinate of the event.
6926          * @return {Number}
6927          */
6928         getPageY : function(){
6929             return this.xy[1];
6930         },
6931
6932         /**
6933          * Gets the time of the event.
6934          * @return {Number}
6935          */
6936         getTime : function(){
6937             if(this.browserEvent){
6938                 return E.getTime(this.browserEvent);
6939             }
6940             return null;
6941         },
6942
6943         /**
6944          * Gets the page coordinates of the event.
6945          * @return {Array} The xy values like [x, y]
6946          */
6947         getXY : function(){
6948             return this.xy;
6949         },
6950
6951         /**
6952          * Gets the target for the event.
6953          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6954          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6955                 search as a number or element (defaults to 10 || document.body)
6956          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6957          * @return {HTMLelement}
6958          */
6959         getTarget : function(selector, maxDepth, returnEl){
6960             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6961         },
6962         /**
6963          * Gets the related target.
6964          * @return {HTMLElement}
6965          */
6966         getRelatedTarget : function(){
6967             if(this.browserEvent){
6968                 return E.getRelatedTarget(this.browserEvent);
6969             }
6970             return null;
6971         },
6972
6973         /**
6974          * Normalizes mouse wheel delta across browsers
6975          * @return {Number} The delta
6976          */
6977         getWheelDelta : function(){
6978             var e = this.browserEvent;
6979             var delta = 0;
6980             if(e.wheelDelta){ /* IE/Opera. */
6981                 delta = e.wheelDelta/120;
6982             }else if(e.detail){ /* Mozilla case. */
6983                 delta = -e.detail/3;
6984             }
6985             return delta;
6986         },
6987
6988         /**
6989          * Returns true if the control, meta, shift or alt key was pressed during this event.
6990          * @return {Boolean}
6991          */
6992         hasModifier : function(){
6993             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6994         },
6995
6996         /**
6997          * Returns true if the target of this event equals el or is a child of el
6998          * @param {String/HTMLElement/Element} el
6999          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
7000          * @return {Boolean}
7001          */
7002         within : function(el, related){
7003             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7004             return t && Roo.fly(el).contains(t);
7005         },
7006
7007         getPoint : function(){
7008             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7009         }
7010     };
7011
7012     return new Roo.EventObjectImpl();
7013 }();
7014             
7015     /*
7016  * Based on:
7017  * Ext JS Library 1.1.1
7018  * Copyright(c) 2006-2007, Ext JS, LLC.
7019  *
7020  * Originally Released Under LGPL - original licence link has changed is not relivant.
7021  *
7022  * Fork - LGPL
7023  * <script type="text/javascript">
7024  */
7025
7026  
7027 // was in Composite Element!??!?!
7028  
7029 (function(){
7030     var D = Roo.lib.Dom;
7031     var E = Roo.lib.Event;
7032     var A = Roo.lib.Anim;
7033
7034     // local style camelizing for speed
7035     var propCache = {};
7036     var camelRe = /(-[a-z])/gi;
7037     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7038     var view = document.defaultView;
7039
7040 /**
7041  * @class Roo.Element
7042  * Represents an Element in the DOM.<br><br>
7043  * Usage:<br>
7044 <pre><code>
7045 var el = Roo.get("my-div");
7046
7047 // or with getEl
7048 var el = getEl("my-div");
7049
7050 // or with a DOM element
7051 var el = Roo.get(myDivElement);
7052 </code></pre>
7053  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7054  * each call instead of constructing a new one.<br><br>
7055  * <b>Animations</b><br />
7056  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7057  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7058 <pre>
7059 Option    Default   Description
7060 --------- --------  ---------------------------------------------
7061 duration  .35       The duration of the animation in seconds
7062 easing    easeOut   The YUI easing method
7063 callback  none      A function to execute when the anim completes
7064 scope     this      The scope (this) of the callback function
7065 </pre>
7066 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7067 * manipulate the animation. Here's an example:
7068 <pre><code>
7069 var el = Roo.get("my-div");
7070
7071 // no animation
7072 el.setWidth(100);
7073
7074 // default animation
7075 el.setWidth(100, true);
7076
7077 // animation with some options set
7078 el.setWidth(100, {
7079     duration: 1,
7080     callback: this.foo,
7081     scope: this
7082 });
7083
7084 // using the "anim" property to get the Anim object
7085 var opt = {
7086     duration: 1,
7087     callback: this.foo,
7088     scope: this
7089 };
7090 el.setWidth(100, opt);
7091 ...
7092 if(opt.anim.isAnimated()){
7093     opt.anim.stop();
7094 }
7095 </code></pre>
7096 * <b> Composite (Collections of) Elements</b><br />
7097  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7098  * @constructor Create a new Element directly.
7099  * @param {String/HTMLElement} element
7100  * @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).
7101  */
7102     Roo.Element = function(element, forceNew){
7103         var dom = typeof element == "string" ?
7104                 document.getElementById(element) : element;
7105         if(!dom){ // invalid id/element
7106             return null;
7107         }
7108         var id = dom.id;
7109         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7110             return Roo.Element.cache[id];
7111         }
7112
7113         /**
7114          * The DOM element
7115          * @type HTMLElement
7116          */
7117         this.dom = dom;
7118
7119         /**
7120          * The DOM element ID
7121          * @type String
7122          */
7123         this.id = id || Roo.id(dom);
7124     };
7125
7126     var El = Roo.Element;
7127
7128     El.prototype = {
7129         /**
7130          * The element's default display mode  (defaults to "") 
7131          * @type String
7132          */
7133         originalDisplay : "",
7134
7135         
7136         // note this is overridden in BS version..
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         if (key === 'length') {
13308             return null;
13309         }
13310         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13311         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13312     },
13313     
13314 /**
13315  * Returns the item at the specified index.
13316  * @param {Number} index The index of the item.
13317  * @return {Object}
13318  */
13319     itemAt : function(index){
13320         return this.items[index];
13321     },
13322     
13323 /**
13324  * Returns the item associated with the passed key.
13325  * @param {String/Number} key The key of the item.
13326  * @return {Object} The item associated with the passed key.
13327  */
13328     key : function(key){
13329         return this.map[key];
13330     },
13331    
13332 /**
13333  * Returns true if the collection contains the passed Object as an item.
13334  * @param {Object} o  The Object to look for in the collection.
13335  * @return {Boolean} True if the collection contains the Object as an item.
13336  */
13337     contains : function(o){
13338         return this.indexOf(o) != -1;
13339     },
13340    
13341 /**
13342  * Returns true if the collection contains the passed Object as a key.
13343  * @param {String} key The key to look for in the collection.
13344  * @return {Boolean} True if the collection contains the Object as a key.
13345  */
13346     containsKey : function(key){
13347         return typeof this.map[key] != "undefined";
13348     },
13349    
13350 /**
13351  * Removes all items from the collection.
13352  */
13353     clear : function(){
13354         this.length = 0;
13355         this.items = [];
13356         this.keys = [];
13357         this.map = {};
13358         this.fireEvent("clear");
13359     },
13360    
13361 /**
13362  * Returns the first item in the collection.
13363  * @return {Object} the first item in the collection..
13364  */
13365     first : function(){
13366         return this.items[0]; 
13367     },
13368    
13369 /**
13370  * Returns the last item in the collection.
13371  * @return {Object} the last item in the collection..
13372  */
13373     last : function(){
13374         return this.items[this.length-1];   
13375     },
13376     
13377     _sort : function(property, dir, fn){
13378         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13379         fn = fn || function(a, b){
13380             return a-b;
13381         };
13382         var c = [], k = this.keys, items = this.items;
13383         for(var i = 0, len = items.length; i < len; i++){
13384             c[c.length] = {key: k[i], value: items[i], index: i};
13385         }
13386         c.sort(function(a, b){
13387             var v = fn(a[property], b[property]) * dsc;
13388             if(v == 0){
13389                 v = (a.index < b.index ? -1 : 1);
13390             }
13391             return v;
13392         });
13393         for(var i = 0, len = c.length; i < len; i++){
13394             items[i] = c[i].value;
13395             k[i] = c[i].key;
13396         }
13397         this.fireEvent("sort", this);
13398     },
13399     
13400     /**
13401      * Sorts this collection with the passed comparison function
13402      * @param {String} direction (optional) "ASC" or "DESC"
13403      * @param {Function} fn (optional) comparison function
13404      */
13405     sort : function(dir, fn){
13406         this._sort("value", dir, fn);
13407     },
13408     
13409     /**
13410      * Sorts this collection by keys
13411      * @param {String} direction (optional) "ASC" or "DESC"
13412      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13413      */
13414     keySort : function(dir, fn){
13415         this._sort("key", dir, fn || function(a, b){
13416             return String(a).toUpperCase()-String(b).toUpperCase();
13417         });
13418     },
13419     
13420     /**
13421      * Returns a range of items in this collection
13422      * @param {Number} startIndex (optional) defaults to 0
13423      * @param {Number} endIndex (optional) default to the last item
13424      * @return {Array} An array of items
13425      */
13426     getRange : function(start, end){
13427         var items = this.items;
13428         if(items.length < 1){
13429             return [];
13430         }
13431         start = start || 0;
13432         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13433         var r = [];
13434         if(start <= end){
13435             for(var i = start; i <= end; i++) {
13436                     r[r.length] = items[i];
13437             }
13438         }else{
13439             for(var i = start; i >= end; i--) {
13440                     r[r.length] = items[i];
13441             }
13442         }
13443         return r;
13444     },
13445         
13446     /**
13447      * Filter the <i>objects</i> in this collection by a specific property. 
13448      * Returns a new collection that has been filtered.
13449      * @param {String} property A property on your objects
13450      * @param {String/RegExp} value Either string that the property values 
13451      * should start with or a RegExp to test against the property
13452      * @return {MixedCollection} The new filtered collection
13453      */
13454     filter : function(property, value){
13455         if(!value.exec){ // not a regex
13456             value = String(value);
13457             if(value.length == 0){
13458                 return this.clone();
13459             }
13460             value = new RegExp("^" + Roo.escapeRe(value), "i");
13461         }
13462         return this.filterBy(function(o){
13463             return o && value.test(o[property]);
13464         });
13465         },
13466     
13467     /**
13468      * Filter by a function. * Returns a new collection that has been filtered.
13469      * The passed function will be called with each 
13470      * object in the collection. If the function returns true, the value is included 
13471      * otherwise it is filtered.
13472      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13473      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13474      * @return {MixedCollection} The new filtered collection
13475      */
13476     filterBy : function(fn, scope){
13477         var r = new Roo.util.MixedCollection();
13478         r.getKey = this.getKey;
13479         var k = this.keys, it = this.items;
13480         for(var i = 0, len = it.length; i < len; i++){
13481             if(fn.call(scope||this, it[i], k[i])){
13482                                 r.add(k[i], it[i]);
13483                         }
13484         }
13485         return r;
13486     },
13487     
13488     /**
13489      * Creates a duplicate of this collection
13490      * @return {MixedCollection}
13491      */
13492     clone : function(){
13493         var r = new Roo.util.MixedCollection();
13494         var k = this.keys, it = this.items;
13495         for(var i = 0, len = it.length; i < len; i++){
13496             r.add(k[i], it[i]);
13497         }
13498         r.getKey = this.getKey;
13499         return r;
13500     }
13501 });
13502 /**
13503  * Returns the item associated with the passed key or index.
13504  * @method
13505  * @param {String/Number} key The key or index of the item.
13506  * @return {Object} The item associated with the passed key.
13507  */
13508 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13509  * Based on:
13510  * Ext JS Library 1.1.1
13511  * Copyright(c) 2006-2007, Ext JS, LLC.
13512  *
13513  * Originally Released Under LGPL - original licence link has changed is not relivant.
13514  *
13515  * Fork - LGPL
13516  * <script type="text/javascript">
13517  */
13518 /**
13519  * @class Roo.util.JSON
13520  * Modified version of Douglas Crockford"s json.js that doesn"t
13521  * mess with the Object prototype 
13522  * http://www.json.org/js.html
13523  * @singleton
13524  */
13525 Roo.util.JSON = new (function(){
13526     var useHasOwn = {}.hasOwnProperty ? true : false;
13527     
13528     // crashes Safari in some instances
13529     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13530     
13531     var pad = function(n) {
13532         return n < 10 ? "0" + n : n;
13533     };
13534     
13535     var m = {
13536         "\b": '\\b',
13537         "\t": '\\t',
13538         "\n": '\\n',
13539         "\f": '\\f',
13540         "\r": '\\r',
13541         '"' : '\\"',
13542         "\\": '\\\\'
13543     };
13544
13545     var encodeString = function(s){
13546         if (/["\\\x00-\x1f]/.test(s)) {
13547             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13548                 var c = m[b];
13549                 if(c){
13550                     return c;
13551                 }
13552                 c = b.charCodeAt();
13553                 return "\\u00" +
13554                     Math.floor(c / 16).toString(16) +
13555                     (c % 16).toString(16);
13556             }) + '"';
13557         }
13558         return '"' + s + '"';
13559     };
13560     
13561     var encodeArray = function(o){
13562         var a = ["["], b, i, l = o.length, v;
13563             for (i = 0; i < l; i += 1) {
13564                 v = o[i];
13565                 switch (typeof v) {
13566                     case "undefined":
13567                     case "function":
13568                     case "unknown":
13569                         break;
13570                     default:
13571                         if (b) {
13572                             a.push(',');
13573                         }
13574                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13575                         b = true;
13576                 }
13577             }
13578             a.push("]");
13579             return a.join("");
13580     };
13581     
13582     var encodeDate = function(o){
13583         return '"' + o.getFullYear() + "-" +
13584                 pad(o.getMonth() + 1) + "-" +
13585                 pad(o.getDate()) + "T" +
13586                 pad(o.getHours()) + ":" +
13587                 pad(o.getMinutes()) + ":" +
13588                 pad(o.getSeconds()) + '"';
13589     };
13590     
13591     /**
13592      * Encodes an Object, Array or other value
13593      * @param {Mixed} o The variable to encode
13594      * @return {String} The JSON string
13595      */
13596     this.encode = function(o)
13597     {
13598         // should this be extended to fully wrap stringify..
13599         
13600         if(typeof o == "undefined" || o === null){
13601             return "null";
13602         }else if(o instanceof Array){
13603             return encodeArray(o);
13604         }else if(o instanceof Date){
13605             return encodeDate(o);
13606         }else if(typeof o == "string"){
13607             return encodeString(o);
13608         }else if(typeof o == "number"){
13609             return isFinite(o) ? String(o) : "null";
13610         }else if(typeof o == "boolean"){
13611             return String(o);
13612         }else {
13613             var a = ["{"], b, i, v;
13614             for (i in o) {
13615                 if(!useHasOwn || o.hasOwnProperty(i)) {
13616                     v = o[i];
13617                     switch (typeof v) {
13618                     case "undefined":
13619                     case "function":
13620                     case "unknown":
13621                         break;
13622                     default:
13623                         if(b){
13624                             a.push(',');
13625                         }
13626                         a.push(this.encode(i), ":",
13627                                 v === null ? "null" : this.encode(v));
13628                         b = true;
13629                     }
13630                 }
13631             }
13632             a.push("}");
13633             return a.join("");
13634         }
13635     };
13636     
13637     /**
13638      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13639      * @param {String} json The JSON string
13640      * @return {Object} The resulting object
13641      */
13642     this.decode = function(json){
13643         
13644         return  /** eval:var:json */ eval("(" + json + ')');
13645     };
13646 })();
13647 /** 
13648  * Shorthand for {@link Roo.util.JSON#encode}
13649  * @member Roo encode 
13650  * @method */
13651 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13652 /** 
13653  * Shorthand for {@link Roo.util.JSON#decode}
13654  * @member Roo decode 
13655  * @method */
13656 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13657 /*
13658  * Based on:
13659  * Ext JS Library 1.1.1
13660  * Copyright(c) 2006-2007, Ext JS, LLC.
13661  *
13662  * Originally Released Under LGPL - original licence link has changed is not relivant.
13663  *
13664  * Fork - LGPL
13665  * <script type="text/javascript">
13666  */
13667  
13668 /**
13669  * @class Roo.util.Format
13670  * Reusable data formatting functions
13671  * @singleton
13672  */
13673 Roo.util.Format = function(){
13674     var trimRe = /^\s+|\s+$/g;
13675     return {
13676         /**
13677          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13678          * @param {String} value The string to truncate
13679          * @param {Number} length The maximum length to allow before truncating
13680          * @return {String} The converted text
13681          */
13682         ellipsis : function(value, len){
13683             if(value && value.length > len){
13684                 return value.substr(0, len-3)+"...";
13685             }
13686             return value;
13687         },
13688
13689         /**
13690          * Checks a reference and converts it to empty string if it is undefined
13691          * @param {Mixed} value Reference to check
13692          * @return {Mixed} Empty string if converted, otherwise the original value
13693          */
13694         undef : function(value){
13695             return typeof value != "undefined" ? value : "";
13696         },
13697
13698         /**
13699          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13700          * @param {String} value The string to encode
13701          * @return {String} The encoded text
13702          */
13703         htmlEncode : function(value){
13704             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13705         },
13706
13707         /**
13708          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13709          * @param {String} value The string to decode
13710          * @return {String} The decoded text
13711          */
13712         htmlDecode : function(value){
13713             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13714         },
13715
13716         /**
13717          * Trims any whitespace from either side of a string
13718          * @param {String} value The text to trim
13719          * @return {String} The trimmed text
13720          */
13721         trim : function(value){
13722             return String(value).replace(trimRe, "");
13723         },
13724
13725         /**
13726          * Returns a substring from within an original string
13727          * @param {String} value The original text
13728          * @param {Number} start The start index of the substring
13729          * @param {Number} length The length of the substring
13730          * @return {String} The substring
13731          */
13732         substr : function(value, start, length){
13733             return String(value).substr(start, length);
13734         },
13735
13736         /**
13737          * Converts a string to all lower case letters
13738          * @param {String} value The text to convert
13739          * @return {String} The converted text
13740          */
13741         lowercase : function(value){
13742             return String(value).toLowerCase();
13743         },
13744
13745         /**
13746          * Converts a string to all upper case letters
13747          * @param {String} value The text to convert
13748          * @return {String} The converted text
13749          */
13750         uppercase : function(value){
13751             return String(value).toUpperCase();
13752         },
13753
13754         /**
13755          * Converts the first character only of a string to upper case
13756          * @param {String} value The text to convert
13757          * @return {String} The converted text
13758          */
13759         capitalize : function(value){
13760             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13761         },
13762
13763         // private
13764         call : function(value, fn){
13765             if(arguments.length > 2){
13766                 var args = Array.prototype.slice.call(arguments, 2);
13767                 args.unshift(value);
13768                  
13769                 return /** eval:var:value */  eval(fn).apply(window, args);
13770             }else{
13771                 /** eval:var:value */
13772                 return /** eval:var:value */ eval(fn).call(window, value);
13773             }
13774         },
13775
13776        
13777         /**
13778          * safer version of Math.toFixed..??/
13779          * @param {Number/String} value The numeric value to format
13780          * @param {Number/String} value Decimal places 
13781          * @return {String} The formatted currency string
13782          */
13783         toFixed : function(v, n)
13784         {
13785             // why not use to fixed - precision is buggered???
13786             if (!n) {
13787                 return Math.round(v-0);
13788             }
13789             var fact = Math.pow(10,n+1);
13790             v = (Math.round((v-0)*fact))/fact;
13791             var z = (''+fact).substring(2);
13792             if (v == Math.floor(v)) {
13793                 return Math.floor(v) + '.' + z;
13794             }
13795             
13796             // now just padd decimals..
13797             var ps = String(v).split('.');
13798             var fd = (ps[1] + z);
13799             var r = fd.substring(0,n); 
13800             var rm = fd.substring(n); 
13801             if (rm < 5) {
13802                 return ps[0] + '.' + r;
13803             }
13804             r*=1; // turn it into a number;
13805             r++;
13806             if (String(r).length != n) {
13807                 ps[0]*=1;
13808                 ps[0]++;
13809                 r = String(r).substring(1); // chop the end off.
13810             }
13811             
13812             return ps[0] + '.' + r;
13813              
13814         },
13815         
13816         /**
13817          * Format a number as US currency
13818          * @param {Number/String} value The numeric value to format
13819          * @return {String} The formatted currency string
13820          */
13821         usMoney : function(v){
13822             return '$' + Roo.util.Format.number(v);
13823         },
13824         
13825         /**
13826          * Format a number
13827          * eventually this should probably emulate php's number_format
13828          * @param {Number/String} value The numeric value to format
13829          * @param {Number} decimals number of decimal places
13830          * @param {String} delimiter for thousands (default comma)
13831          * @return {String} The formatted currency string
13832          */
13833         number : function(v, decimals, thousandsDelimiter)
13834         {
13835             // multiply and round.
13836             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13837             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13838             
13839             var mul = Math.pow(10, decimals);
13840             var zero = String(mul).substring(1);
13841             v = (Math.round((v-0)*mul))/mul;
13842             
13843             // if it's '0' number.. then
13844             
13845             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13846             v = String(v);
13847             var ps = v.split('.');
13848             var whole = ps[0];
13849             
13850             var r = /(\d+)(\d{3})/;
13851             // add comma's
13852             
13853             if(thousandsDelimiter.length != 0) {
13854                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13855             } 
13856             
13857             var sub = ps[1] ?
13858                     // has decimals..
13859                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13860                     // does not have decimals
13861                     (decimals ? ('.' + zero) : '');
13862             
13863             
13864             return whole + sub ;
13865         },
13866         
13867         /**
13868          * Parse a value into a formatted date using the specified format pattern.
13869          * @param {Mixed} value The value to format
13870          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13871          * @return {String} The formatted date string
13872          */
13873         date : function(v, format){
13874             if(!v){
13875                 return "";
13876             }
13877             if(!(v instanceof Date)){
13878                 v = new Date(Date.parse(v));
13879             }
13880             return v.dateFormat(format || Roo.util.Format.defaults.date);
13881         },
13882
13883         /**
13884          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13885          * @param {String} format Any valid date format string
13886          * @return {Function} The date formatting function
13887          */
13888         dateRenderer : function(format){
13889             return function(v){
13890                 return Roo.util.Format.date(v, format);  
13891             };
13892         },
13893
13894         // private
13895         stripTagsRE : /<\/?[^>]+>/gi,
13896         
13897         /**
13898          * Strips all HTML tags
13899          * @param {Mixed} value The text from which to strip tags
13900          * @return {String} The stripped text
13901          */
13902         stripTags : function(v){
13903             return !v ? v : String(v).replace(this.stripTagsRE, "");
13904         }
13905     };
13906 }();
13907 Roo.util.Format.defaults = {
13908     date : 'd/M/Y'
13909 };/*
13910  * Based on:
13911  * Ext JS Library 1.1.1
13912  * Copyright(c) 2006-2007, Ext JS, LLC.
13913  *
13914  * Originally Released Under LGPL - original licence link has changed is not relivant.
13915  *
13916  * Fork - LGPL
13917  * <script type="text/javascript">
13918  */
13919
13920
13921  
13922
13923 /**
13924  * @class Roo.MasterTemplate
13925  * @extends Roo.Template
13926  * Provides a template that can have child templates. The syntax is:
13927 <pre><code>
13928 var t = new Roo.MasterTemplate(
13929         '&lt;select name="{name}"&gt;',
13930                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13931         '&lt;/select&gt;'
13932 );
13933 t.add('options', {value: 'foo', text: 'bar'});
13934 // or you can add multiple child elements in one shot
13935 t.addAll('options', [
13936     {value: 'foo', text: 'bar'},
13937     {value: 'foo2', text: 'bar2'},
13938     {value: 'foo3', text: 'bar3'}
13939 ]);
13940 // then append, applying the master template values
13941 t.append('my-form', {name: 'my-select'});
13942 </code></pre>
13943 * A name attribute for the child template is not required if you have only one child
13944 * template or you want to refer to them by index.
13945  */
13946 Roo.MasterTemplate = function(){
13947     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13948     this.originalHtml = this.html;
13949     var st = {};
13950     var m, re = this.subTemplateRe;
13951     re.lastIndex = 0;
13952     var subIndex = 0;
13953     while(m = re.exec(this.html)){
13954         var name = m[1], content = m[2];
13955         st[subIndex] = {
13956             name: name,
13957             index: subIndex,
13958             buffer: [],
13959             tpl : new Roo.Template(content)
13960         };
13961         if(name){
13962             st[name] = st[subIndex];
13963         }
13964         st[subIndex].tpl.compile();
13965         st[subIndex].tpl.call = this.call.createDelegate(this);
13966         subIndex++;
13967     }
13968     this.subCount = subIndex;
13969     this.subs = st;
13970 };
13971 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13972     /**
13973     * The regular expression used to match sub templates
13974     * @type RegExp
13975     * @property
13976     */
13977     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13978
13979     /**
13980      * Applies the passed values to a child template.
13981      * @param {String/Number} name (optional) The name or index of the child template
13982      * @param {Array/Object} values The values to be applied to the template
13983      * @return {MasterTemplate} this
13984      */
13985      add : function(name, values){
13986         if(arguments.length == 1){
13987             values = arguments[0];
13988             name = 0;
13989         }
13990         var s = this.subs[name];
13991         s.buffer[s.buffer.length] = s.tpl.apply(values);
13992         return this;
13993     },
13994
13995     /**
13996      * Applies all the passed values to a child template.
13997      * @param {String/Number} name (optional) The name or index of the child template
13998      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13999      * @param {Boolean} reset (optional) True to reset the template first
14000      * @return {MasterTemplate} this
14001      */
14002     fill : function(name, values, reset){
14003         var a = arguments;
14004         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
14005             values = a[0];
14006             name = 0;
14007             reset = a[1];
14008         }
14009         if(reset){
14010             this.reset();
14011         }
14012         for(var i = 0, len = values.length; i < len; i++){
14013             this.add(name, values[i]);
14014         }
14015         return this;
14016     },
14017
14018     /**
14019      * Resets the template for reuse
14020      * @return {MasterTemplate} this
14021      */
14022      reset : function(){
14023         var s = this.subs;
14024         for(var i = 0; i < this.subCount; i++){
14025             s[i].buffer = [];
14026         }
14027         return this;
14028     },
14029
14030     applyTemplate : function(values){
14031         var s = this.subs;
14032         var replaceIndex = -1;
14033         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14034             return s[++replaceIndex].buffer.join("");
14035         });
14036         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14037     },
14038
14039     apply : function(){
14040         return this.applyTemplate.apply(this, arguments);
14041     },
14042
14043     compile : function(){return this;}
14044 });
14045
14046 /**
14047  * Alias for fill().
14048  * @method
14049  */
14050 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14051  /**
14052  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14053  * var tpl = Roo.MasterTemplate.from('element-id');
14054  * @param {String/HTMLElement} el
14055  * @param {Object} config
14056  * @static
14057  */
14058 Roo.MasterTemplate.from = function(el, config){
14059     el = Roo.getDom(el);
14060     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14061 };/*
14062  * Based on:
14063  * Ext JS Library 1.1.1
14064  * Copyright(c) 2006-2007, Ext JS, LLC.
14065  *
14066  * Originally Released Under LGPL - original licence link has changed is not relivant.
14067  *
14068  * Fork - LGPL
14069  * <script type="text/javascript">
14070  */
14071
14072  
14073 /**
14074  * @class Roo.util.CSS
14075  * Utility class for manipulating CSS rules
14076  * @singleton
14077  */
14078 Roo.util.CSS = function(){
14079         var rules = null;
14080         var doc = document;
14081
14082     var camelRe = /(-[a-z])/gi;
14083     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14084
14085    return {
14086    /**
14087     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14088     * tag and appended to the HEAD of the document.
14089     * @param {String|Object} cssText The text containing the css rules
14090     * @param {String} id An id to add to the stylesheet for later removal
14091     * @return {StyleSheet}
14092     */
14093     createStyleSheet : function(cssText, id){
14094         var ss;
14095         var head = doc.getElementsByTagName("head")[0];
14096         var nrules = doc.createElement("style");
14097         nrules.setAttribute("type", "text/css");
14098         if(id){
14099             nrules.setAttribute("id", id);
14100         }
14101         if (typeof(cssText) != 'string') {
14102             // support object maps..
14103             // not sure if this a good idea.. 
14104             // perhaps it should be merged with the general css handling
14105             // and handle js style props.
14106             var cssTextNew = [];
14107             for(var n in cssText) {
14108                 var citems = [];
14109                 for(var k in cssText[n]) {
14110                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14111                 }
14112                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14113                 
14114             }
14115             cssText = cssTextNew.join("\n");
14116             
14117         }
14118        
14119        
14120        if(Roo.isIE){
14121            head.appendChild(nrules);
14122            ss = nrules.styleSheet;
14123            ss.cssText = cssText;
14124        }else{
14125            try{
14126                 nrules.appendChild(doc.createTextNode(cssText));
14127            }catch(e){
14128                nrules.cssText = cssText; 
14129            }
14130            head.appendChild(nrules);
14131            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14132        }
14133        this.cacheStyleSheet(ss);
14134        return ss;
14135    },
14136
14137    /**
14138     * Removes a style or link tag by id
14139     * @param {String} id The id of the tag
14140     */
14141    removeStyleSheet : function(id){
14142        var existing = doc.getElementById(id);
14143        if(existing){
14144            existing.parentNode.removeChild(existing);
14145        }
14146    },
14147
14148    /**
14149     * Dynamically swaps an existing stylesheet reference for a new one
14150     * @param {String} id The id of an existing link tag to remove
14151     * @param {String} url The href of the new stylesheet to include
14152     */
14153    swapStyleSheet : function(id, url){
14154        this.removeStyleSheet(id);
14155        var ss = doc.createElement("link");
14156        ss.setAttribute("rel", "stylesheet");
14157        ss.setAttribute("type", "text/css");
14158        ss.setAttribute("id", id);
14159        ss.setAttribute("href", url);
14160        doc.getElementsByTagName("head")[0].appendChild(ss);
14161    },
14162    
14163    /**
14164     * Refresh the rule cache if you have dynamically added stylesheets
14165     * @return {Object} An object (hash) of rules indexed by selector
14166     */
14167    refreshCache : function(){
14168        return this.getRules(true);
14169    },
14170
14171    // private
14172    cacheStyleSheet : function(stylesheet){
14173        if(!rules){
14174            rules = {};
14175        }
14176        try{// try catch for cross domain access issue
14177            var ssRules = stylesheet.cssRules || stylesheet.rules;
14178            for(var j = ssRules.length-1; j >= 0; --j){
14179                rules[ssRules[j].selectorText] = ssRules[j];
14180            }
14181        }catch(e){}
14182    },
14183    
14184    /**
14185     * Gets all css rules for the document
14186     * @param {Boolean} refreshCache true to refresh the internal cache
14187     * @return {Object} An object (hash) of rules indexed by selector
14188     */
14189    getRules : function(refreshCache){
14190                 if(rules == null || refreshCache){
14191                         rules = {};
14192                         var ds = doc.styleSheets;
14193                         for(var i =0, len = ds.length; i < len; i++){
14194                             try{
14195                         this.cacheStyleSheet(ds[i]);
14196                     }catch(e){} 
14197                 }
14198                 }
14199                 return rules;
14200         },
14201         
14202         /**
14203     * Gets an an individual CSS rule by selector(s)
14204     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14205     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14206     * @return {CSSRule} The CSS rule or null if one is not found
14207     */
14208    getRule : function(selector, refreshCache){
14209                 var rs = this.getRules(refreshCache);
14210                 if(!(selector instanceof Array)){
14211                     return rs[selector];
14212                 }
14213                 for(var i = 0; i < selector.length; i++){
14214                         if(rs[selector[i]]){
14215                                 return rs[selector[i]];
14216                         }
14217                 }
14218                 return null;
14219         },
14220         
14221         
14222         /**
14223     * Updates a rule property
14224     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14225     * @param {String} property The css property
14226     * @param {String} value The new value for the property
14227     * @return {Boolean} true If a rule was found and updated
14228     */
14229    updateRule : function(selector, property, value){
14230                 if(!(selector instanceof Array)){
14231                         var rule = this.getRule(selector);
14232                         if(rule){
14233                                 rule.style[property.replace(camelRe, camelFn)] = value;
14234                                 return true;
14235                         }
14236                 }else{
14237                         for(var i = 0; i < selector.length; i++){
14238                                 if(this.updateRule(selector[i], property, value)){
14239                                         return true;
14240                                 }
14241                         }
14242                 }
14243                 return false;
14244         }
14245    };   
14246 }();/*
14247  * Based on:
14248  * Ext JS Library 1.1.1
14249  * Copyright(c) 2006-2007, Ext JS, LLC.
14250  *
14251  * Originally Released Under LGPL - original licence link has changed is not relivant.
14252  *
14253  * Fork - LGPL
14254  * <script type="text/javascript">
14255  */
14256
14257  
14258
14259 /**
14260  * @class Roo.util.ClickRepeater
14261  * @extends Roo.util.Observable
14262  * 
14263  * A wrapper class which can be applied to any element. Fires a "click" event while the
14264  * mouse is pressed. The interval between firings may be specified in the config but
14265  * defaults to 10 milliseconds.
14266  * 
14267  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14268  * 
14269  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14270  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14271  * Similar to an autorepeat key delay.
14272  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14273  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14274  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14275  *           "interval" and "delay" are ignored. "immediate" is honored.
14276  * @cfg {Boolean} preventDefault True to prevent the default click event
14277  * @cfg {Boolean} stopDefault True to stop the default click event
14278  * 
14279  * @history
14280  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14281  *     2007-02-02 jvs Renamed to ClickRepeater
14282  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14283  *
14284  *  @constructor
14285  * @param {String/HTMLElement/Element} el The element to listen on
14286  * @param {Object} config
14287  **/
14288 Roo.util.ClickRepeater = function(el, config)
14289 {
14290     this.el = Roo.get(el);
14291     this.el.unselectable();
14292
14293     Roo.apply(this, config);
14294
14295     this.addEvents({
14296     /**
14297      * @event mousedown
14298      * Fires when the mouse button is depressed.
14299      * @param {Roo.util.ClickRepeater} this
14300      */
14301         "mousedown" : true,
14302     /**
14303      * @event click
14304      * Fires on a specified interval during the time the element is pressed.
14305      * @param {Roo.util.ClickRepeater} this
14306      */
14307         "click" : true,
14308     /**
14309      * @event mouseup
14310      * Fires when the mouse key is released.
14311      * @param {Roo.util.ClickRepeater} this
14312      */
14313         "mouseup" : true
14314     });
14315
14316     this.el.on("mousedown", this.handleMouseDown, this);
14317     if(this.preventDefault || this.stopDefault){
14318         this.el.on("click", function(e){
14319             if(this.preventDefault){
14320                 e.preventDefault();
14321             }
14322             if(this.stopDefault){
14323                 e.stopEvent();
14324             }
14325         }, this);
14326     }
14327
14328     // allow inline handler
14329     if(this.handler){
14330         this.on("click", this.handler,  this.scope || this);
14331     }
14332
14333     Roo.util.ClickRepeater.superclass.constructor.call(this);
14334 };
14335
14336 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14337     interval : 20,
14338     delay: 250,
14339     preventDefault : true,
14340     stopDefault : false,
14341     timer : 0,
14342
14343     // private
14344     handleMouseDown : function(){
14345         clearTimeout(this.timer);
14346         this.el.blur();
14347         if(this.pressClass){
14348             this.el.addClass(this.pressClass);
14349         }
14350         this.mousedownTime = new Date();
14351
14352         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14353         this.el.on("mouseout", this.handleMouseOut, this);
14354
14355         this.fireEvent("mousedown", this);
14356         this.fireEvent("click", this);
14357         
14358         this.timer = this.click.defer(this.delay || this.interval, this);
14359     },
14360
14361     // private
14362     click : function(){
14363         this.fireEvent("click", this);
14364         this.timer = this.click.defer(this.getInterval(), this);
14365     },
14366
14367     // private
14368     getInterval: function(){
14369         if(!this.accelerate){
14370             return this.interval;
14371         }
14372         var pressTime = this.mousedownTime.getElapsed();
14373         if(pressTime < 500){
14374             return 400;
14375         }else if(pressTime < 1700){
14376             return 320;
14377         }else if(pressTime < 2600){
14378             return 250;
14379         }else if(pressTime < 3500){
14380             return 180;
14381         }else if(pressTime < 4400){
14382             return 140;
14383         }else if(pressTime < 5300){
14384             return 80;
14385         }else if(pressTime < 6200){
14386             return 50;
14387         }else{
14388             return 10;
14389         }
14390     },
14391
14392     // private
14393     handleMouseOut : function(){
14394         clearTimeout(this.timer);
14395         if(this.pressClass){
14396             this.el.removeClass(this.pressClass);
14397         }
14398         this.el.on("mouseover", this.handleMouseReturn, this);
14399     },
14400
14401     // private
14402     handleMouseReturn : function(){
14403         this.el.un("mouseover", this.handleMouseReturn);
14404         if(this.pressClass){
14405             this.el.addClass(this.pressClass);
14406         }
14407         this.click();
14408     },
14409
14410     // private
14411     handleMouseUp : function(){
14412         clearTimeout(this.timer);
14413         this.el.un("mouseover", this.handleMouseReturn);
14414         this.el.un("mouseout", this.handleMouseOut);
14415         Roo.get(document).un("mouseup", this.handleMouseUp);
14416         this.el.removeClass(this.pressClass);
14417         this.fireEvent("mouseup", this);
14418     }
14419 });/*
14420  * Based on:
14421  * Ext JS Library 1.1.1
14422  * Copyright(c) 2006-2007, Ext JS, LLC.
14423  *
14424  * Originally Released Under LGPL - original licence link has changed is not relivant.
14425  *
14426  * Fork - LGPL
14427  * <script type="text/javascript">
14428  */
14429
14430  
14431 /**
14432  * @class Roo.KeyNav
14433  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14434  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14435  * way to implement custom navigation schemes for any UI component.</p>
14436  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14437  * pageUp, pageDown, del, home, end.  Usage:</p>
14438  <pre><code>
14439 var nav = new Roo.KeyNav("my-element", {
14440     "left" : function(e){
14441         this.moveLeft(e.ctrlKey);
14442     },
14443     "right" : function(e){
14444         this.moveRight(e.ctrlKey);
14445     },
14446     "enter" : function(e){
14447         this.save();
14448     },
14449     scope : this
14450 });
14451 </code></pre>
14452  * @constructor
14453  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14454  * @param {Object} config The config
14455  */
14456 Roo.KeyNav = function(el, config){
14457     this.el = Roo.get(el);
14458     Roo.apply(this, config);
14459     if(!this.disabled){
14460         this.disabled = true;
14461         this.enable();
14462     }
14463 };
14464
14465 Roo.KeyNav.prototype = {
14466     /**
14467      * @cfg {Boolean} disabled
14468      * True to disable this KeyNav instance (defaults to false)
14469      */
14470     disabled : false,
14471     /**
14472      * @cfg {String} defaultEventAction
14473      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14474      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14475      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14476      */
14477     defaultEventAction: "stopEvent",
14478     /**
14479      * @cfg {Boolean} forceKeyDown
14480      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14481      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14482      * handle keydown instead of keypress.
14483      */
14484     forceKeyDown : false,
14485
14486     // private
14487     prepareEvent : function(e){
14488         var k = e.getKey();
14489         var h = this.keyToHandler[k];
14490         //if(h && this[h]){
14491         //    e.stopPropagation();
14492         //}
14493         if(Roo.isSafari && h && k >= 37 && k <= 40){
14494             e.stopEvent();
14495         }
14496     },
14497
14498     // private
14499     relay : function(e){
14500         var k = e.getKey();
14501         var h = this.keyToHandler[k];
14502         if(h && this[h]){
14503             if(this.doRelay(e, this[h], h) !== true){
14504                 e[this.defaultEventAction]();
14505             }
14506         }
14507     },
14508
14509     // private
14510     doRelay : function(e, h, hname){
14511         return h.call(this.scope || this, e);
14512     },
14513
14514     // possible handlers
14515     enter : false,
14516     left : false,
14517     right : false,
14518     up : false,
14519     down : false,
14520     tab : false,
14521     esc : false,
14522     pageUp : false,
14523     pageDown : false,
14524     del : false,
14525     home : false,
14526     end : false,
14527
14528     // quick lookup hash
14529     keyToHandler : {
14530         37 : "left",
14531         39 : "right",
14532         38 : "up",
14533         40 : "down",
14534         33 : "pageUp",
14535         34 : "pageDown",
14536         46 : "del",
14537         36 : "home",
14538         35 : "end",
14539         13 : "enter",
14540         27 : "esc",
14541         9  : "tab"
14542     },
14543
14544         /**
14545          * Enable this KeyNav
14546          */
14547         enable: function(){
14548                 if(this.disabled){
14549             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14550             // the EventObject will normalize Safari automatically
14551             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14552                 this.el.on("keydown", this.relay,  this);
14553             }else{
14554                 this.el.on("keydown", this.prepareEvent,  this);
14555                 this.el.on("keypress", this.relay,  this);
14556             }
14557                     this.disabled = false;
14558                 }
14559         },
14560
14561         /**
14562          * Disable this KeyNav
14563          */
14564         disable: function(){
14565                 if(!this.disabled){
14566                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14567                 this.el.un("keydown", this.relay);
14568             }else{
14569                 this.el.un("keydown", this.prepareEvent);
14570                 this.el.un("keypress", this.relay);
14571             }
14572                     this.disabled = true;
14573                 }
14574         }
14575 };/*
14576  * Based on:
14577  * Ext JS Library 1.1.1
14578  * Copyright(c) 2006-2007, Ext JS, LLC.
14579  *
14580  * Originally Released Under LGPL - original licence link has changed is not relivant.
14581  *
14582  * Fork - LGPL
14583  * <script type="text/javascript">
14584  */
14585
14586  
14587 /**
14588  * @class Roo.KeyMap
14589  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14590  * The constructor accepts the same config object as defined by {@link #addBinding}.
14591  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14592  * combination it will call the function with this signature (if the match is a multi-key
14593  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14594  * A KeyMap can also handle a string representation of keys.<br />
14595  * Usage:
14596  <pre><code>
14597 // map one key by key code
14598 var map = new Roo.KeyMap("my-element", {
14599     key: 13, // or Roo.EventObject.ENTER
14600     fn: myHandler,
14601     scope: myObject
14602 });
14603
14604 // map multiple keys to one action by string
14605 var map = new Roo.KeyMap("my-element", {
14606     key: "a\r\n\t",
14607     fn: myHandler,
14608     scope: myObject
14609 });
14610
14611 // map multiple keys to multiple actions by strings and array of codes
14612 var map = new Roo.KeyMap("my-element", [
14613     {
14614         key: [10,13],
14615         fn: function(){ alert("Return was pressed"); }
14616     }, {
14617         key: "abc",
14618         fn: function(){ alert('a, b or c was pressed'); }
14619     }, {
14620         key: "\t",
14621         ctrl:true,
14622         shift:true,
14623         fn: function(){ alert('Control + shift + tab was pressed.'); }
14624     }
14625 ]);
14626 </code></pre>
14627  * <b>Note: A KeyMap starts enabled</b>
14628  * @constructor
14629  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14630  * @param {Object} config The config (see {@link #addBinding})
14631  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14632  */
14633 Roo.KeyMap = function(el, config, eventName){
14634     this.el  = Roo.get(el);
14635     this.eventName = eventName || "keydown";
14636     this.bindings = [];
14637     if(config){
14638         this.addBinding(config);
14639     }
14640     this.enable();
14641 };
14642
14643 Roo.KeyMap.prototype = {
14644     /**
14645      * True to stop the event from bubbling and prevent the default browser action if the
14646      * key was handled by the KeyMap (defaults to false)
14647      * @type Boolean
14648      */
14649     stopEvent : false,
14650
14651     /**
14652      * Add a new binding to this KeyMap. The following config object properties are supported:
14653      * <pre>
14654 Property    Type             Description
14655 ----------  ---------------  ----------------------------------------------------------------------
14656 key         String/Array     A single keycode or an array of keycodes to handle
14657 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14658 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14659 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14660 fn          Function         The function to call when KeyMap finds the expected key combination
14661 scope       Object           The scope of the callback function
14662 </pre>
14663      *
14664      * Usage:
14665      * <pre><code>
14666 // Create a KeyMap
14667 var map = new Roo.KeyMap(document, {
14668     key: Roo.EventObject.ENTER,
14669     fn: handleKey,
14670     scope: this
14671 });
14672
14673 //Add a new binding to the existing KeyMap later
14674 map.addBinding({
14675     key: 'abc',
14676     shift: true,
14677     fn: handleKey,
14678     scope: this
14679 });
14680 </code></pre>
14681      * @param {Object/Array} config A single KeyMap config or an array of configs
14682      */
14683         addBinding : function(config){
14684         if(config instanceof Array){
14685             for(var i = 0, len = config.length; i < len; i++){
14686                 this.addBinding(config[i]);
14687             }
14688             return;
14689         }
14690         var keyCode = config.key,
14691             shift = config.shift, 
14692             ctrl = config.ctrl, 
14693             alt = config.alt,
14694             fn = config.fn,
14695             scope = config.scope;
14696         if(typeof keyCode == "string"){
14697             var ks = [];
14698             var keyString = keyCode.toUpperCase();
14699             for(var j = 0, len = keyString.length; j < len; j++){
14700                 ks.push(keyString.charCodeAt(j));
14701             }
14702             keyCode = ks;
14703         }
14704         var keyArray = keyCode instanceof Array;
14705         var handler = function(e){
14706             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14707                 var k = e.getKey();
14708                 if(keyArray){
14709                     for(var i = 0, len = keyCode.length; i < len; i++){
14710                         if(keyCode[i] == k){
14711                           if(this.stopEvent){
14712                               e.stopEvent();
14713                           }
14714                           fn.call(scope || window, k, e);
14715                           return;
14716                         }
14717                     }
14718                 }else{
14719                     if(k == keyCode){
14720                         if(this.stopEvent){
14721                            e.stopEvent();
14722                         }
14723                         fn.call(scope || window, k, e);
14724                     }
14725                 }
14726             }
14727         };
14728         this.bindings.push(handler);  
14729         },
14730
14731     /**
14732      * Shorthand for adding a single key listener
14733      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14734      * following options:
14735      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14736      * @param {Function} fn The function to call
14737      * @param {Object} scope (optional) The scope of the function
14738      */
14739     on : function(key, fn, scope){
14740         var keyCode, shift, ctrl, alt;
14741         if(typeof key == "object" && !(key instanceof Array)){
14742             keyCode = key.key;
14743             shift = key.shift;
14744             ctrl = key.ctrl;
14745             alt = key.alt;
14746         }else{
14747             keyCode = key;
14748         }
14749         this.addBinding({
14750             key: keyCode,
14751             shift: shift,
14752             ctrl: ctrl,
14753             alt: alt,
14754             fn: fn,
14755             scope: scope
14756         })
14757     },
14758
14759     // private
14760     handleKeyDown : function(e){
14761             if(this.enabled){ //just in case
14762             var b = this.bindings;
14763             for(var i = 0, len = b.length; i < len; i++){
14764                 b[i].call(this, e);
14765             }
14766             }
14767         },
14768         
14769         /**
14770          * Returns true if this KeyMap is enabled
14771          * @return {Boolean} 
14772          */
14773         isEnabled : function(){
14774             return this.enabled;  
14775         },
14776         
14777         /**
14778          * Enables this KeyMap
14779          */
14780         enable: function(){
14781                 if(!this.enabled){
14782                     this.el.on(this.eventName, this.handleKeyDown, this);
14783                     this.enabled = true;
14784                 }
14785         },
14786
14787         /**
14788          * Disable this KeyMap
14789          */
14790         disable: function(){
14791                 if(this.enabled){
14792                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14793                     this.enabled = false;
14794                 }
14795         }
14796 };/*
14797  * Based on:
14798  * Ext JS Library 1.1.1
14799  * Copyright(c) 2006-2007, Ext JS, LLC.
14800  *
14801  * Originally Released Under LGPL - original licence link has changed is not relivant.
14802  *
14803  * Fork - LGPL
14804  * <script type="text/javascript">
14805  */
14806
14807  
14808 /**
14809  * @class Roo.util.TextMetrics
14810  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14811  * wide, in pixels, a given block of text will be.
14812  * @singleton
14813  */
14814 Roo.util.TextMetrics = function(){
14815     var shared;
14816     return {
14817         /**
14818          * Measures the size of the specified text
14819          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14820          * that can affect the size of the rendered text
14821          * @param {String} text The text to measure
14822          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14823          * in order to accurately measure the text height
14824          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14825          */
14826         measure : function(el, text, fixedWidth){
14827             if(!shared){
14828                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14829             }
14830             shared.bind(el);
14831             shared.setFixedWidth(fixedWidth || 'auto');
14832             return shared.getSize(text);
14833         },
14834
14835         /**
14836          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14837          * the overhead of multiple calls to initialize the style properties on each measurement.
14838          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14839          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14840          * in order to accurately measure the text height
14841          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14842          */
14843         createInstance : function(el, fixedWidth){
14844             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14845         }
14846     };
14847 }();
14848
14849  
14850
14851 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14852     var ml = new Roo.Element(document.createElement('div'));
14853     document.body.appendChild(ml.dom);
14854     ml.position('absolute');
14855     ml.setLeftTop(-1000, -1000);
14856     ml.hide();
14857
14858     if(fixedWidth){
14859         ml.setWidth(fixedWidth);
14860     }
14861      
14862     var instance = {
14863         /**
14864          * Returns the size of the specified text based on the internal element's style and width properties
14865          * @memberOf Roo.util.TextMetrics.Instance#
14866          * @param {String} text The text to measure
14867          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14868          */
14869         getSize : function(text){
14870             ml.update(text);
14871             var s = ml.getSize();
14872             ml.update('');
14873             return s;
14874         },
14875
14876         /**
14877          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14878          * that can affect the size of the rendered text
14879          * @memberOf Roo.util.TextMetrics.Instance#
14880          * @param {String/HTMLElement} el The element, dom node or id
14881          */
14882         bind : function(el){
14883             ml.setStyle(
14884                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14885             );
14886         },
14887
14888         /**
14889          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14890          * to set a fixed width in order to accurately measure the text height.
14891          * @memberOf Roo.util.TextMetrics.Instance#
14892          * @param {Number} width The width to set on the element
14893          */
14894         setFixedWidth : function(width){
14895             ml.setWidth(width);
14896         },
14897
14898         /**
14899          * Returns the measured width of the specified text
14900          * @memberOf Roo.util.TextMetrics.Instance#
14901          * @param {String} text The text to measure
14902          * @return {Number} width The width in pixels
14903          */
14904         getWidth : function(text){
14905             ml.dom.style.width = 'auto';
14906             return this.getSize(text).width;
14907         },
14908
14909         /**
14910          * Returns the measured height of the specified text.  For multiline text, be sure to call
14911          * {@link #setFixedWidth} if necessary.
14912          * @memberOf Roo.util.TextMetrics.Instance#
14913          * @param {String} text The text to measure
14914          * @return {Number} height The height in pixels
14915          */
14916         getHeight : function(text){
14917             return this.getSize(text).height;
14918         }
14919     };
14920
14921     instance.bind(bindTo);
14922
14923     return instance;
14924 };
14925
14926 // backwards compat
14927 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14928  * Based on:
14929  * Ext JS Library 1.1.1
14930  * Copyright(c) 2006-2007, Ext JS, LLC.
14931  *
14932  * Originally Released Under LGPL - original licence link has changed is not relivant.
14933  *
14934  * Fork - LGPL
14935  * <script type="text/javascript">
14936  */
14937
14938 /**
14939  * @class Roo.state.Provider
14940  * Abstract base class for state provider implementations. This class provides methods
14941  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14942  * Provider interface.
14943  */
14944 Roo.state.Provider = function(){
14945     /**
14946      * @event statechange
14947      * Fires when a state change occurs.
14948      * @param {Provider} this This state provider
14949      * @param {String} key The state key which was changed
14950      * @param {String} value The encoded value for the state
14951      */
14952     this.addEvents({
14953         "statechange": true
14954     });
14955     this.state = {};
14956     Roo.state.Provider.superclass.constructor.call(this);
14957 };
14958 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14959     /**
14960      * Returns the current value for a key
14961      * @param {String} name The key name
14962      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14963      * @return {Mixed} The state data
14964      */
14965     get : function(name, defaultValue){
14966         return typeof this.state[name] == "undefined" ?
14967             defaultValue : this.state[name];
14968     },
14969     
14970     /**
14971      * Clears a value from the state
14972      * @param {String} name The key name
14973      */
14974     clear : function(name){
14975         delete this.state[name];
14976         this.fireEvent("statechange", this, name, null);
14977     },
14978     
14979     /**
14980      * Sets the value for a key
14981      * @param {String} name The key name
14982      * @param {Mixed} value The value to set
14983      */
14984     set : function(name, value){
14985         this.state[name] = value;
14986         this.fireEvent("statechange", this, name, value);
14987     },
14988     
14989     /**
14990      * Decodes a string previously encoded with {@link #encodeValue}.
14991      * @param {String} value The value to decode
14992      * @return {Mixed} The decoded value
14993      */
14994     decodeValue : function(cookie){
14995         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14996         var matches = re.exec(unescape(cookie));
14997         if(!matches || !matches[1]) {
14998             return; // non state cookie
14999         }
15000         var type = matches[1];
15001         var v = matches[2];
15002         switch(type){
15003             case "n":
15004                 return parseFloat(v);
15005             case "d":
15006                 return new Date(Date.parse(v));
15007             case "b":
15008                 return (v == "1");
15009             case "a":
15010                 var all = [];
15011                 var values = v.split("^");
15012                 for(var i = 0, len = values.length; i < len; i++){
15013                     all.push(this.decodeValue(values[i]));
15014                 }
15015                 return all;
15016            case "o":
15017                 var all = {};
15018                 var values = v.split("^");
15019                 for(var i = 0, len = values.length; i < len; i++){
15020                     var kv = values[i].split("=");
15021                     all[kv[0]] = this.decodeValue(kv[1]);
15022                 }
15023                 return all;
15024            default:
15025                 return v;
15026         }
15027     },
15028     
15029     /**
15030      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15031      * @param {Mixed} value The value to encode
15032      * @return {String} The encoded value
15033      */
15034     encodeValue : function(v){
15035         var enc;
15036         if(typeof v == "number"){
15037             enc = "n:" + v;
15038         }else if(typeof v == "boolean"){
15039             enc = "b:" + (v ? "1" : "0");
15040         }else if(v instanceof Date){
15041             enc = "d:" + v.toGMTString();
15042         }else if(v instanceof Array){
15043             var flat = "";
15044             for(var i = 0, len = v.length; i < len; i++){
15045                 flat += this.encodeValue(v[i]);
15046                 if(i != len-1) {
15047                     flat += "^";
15048                 }
15049             }
15050             enc = "a:" + flat;
15051         }else if(typeof v == "object"){
15052             var flat = "";
15053             for(var key in v){
15054                 if(typeof v[key] != "function"){
15055                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15056                 }
15057             }
15058             enc = "o:" + flat.substring(0, flat.length-1);
15059         }else{
15060             enc = "s:" + v;
15061         }
15062         return escape(enc);        
15063     }
15064 });
15065
15066 /*
15067  * Based on:
15068  * Ext JS Library 1.1.1
15069  * Copyright(c) 2006-2007, Ext JS, LLC.
15070  *
15071  * Originally Released Under LGPL - original licence link has changed is not relivant.
15072  *
15073  * Fork - LGPL
15074  * <script type="text/javascript">
15075  */
15076 /**
15077  * @class Roo.state.Manager
15078  * This is the global state manager. By default all components that are "state aware" check this class
15079  * for state information if you don't pass them a custom state provider. In order for this class
15080  * to be useful, it must be initialized with a provider when your application initializes.
15081  <pre><code>
15082 // in your initialization function
15083 init : function(){
15084    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15085    ...
15086    // supposed you have a {@link Roo.BorderLayout}
15087    var layout = new Roo.BorderLayout(...);
15088    layout.restoreState();
15089    // or a {Roo.BasicDialog}
15090    var dialog = new Roo.BasicDialog(...);
15091    dialog.restoreState();
15092  </code></pre>
15093  * @singleton
15094  */
15095 Roo.state.Manager = function(){
15096     var provider = new Roo.state.Provider();
15097     
15098     return {
15099         /**
15100          * Configures the default state provider for your application
15101          * @param {Provider} stateProvider The state provider to set
15102          */
15103         setProvider : function(stateProvider){
15104             provider = stateProvider;
15105         },
15106         
15107         /**
15108          * Returns the current value for a key
15109          * @param {String} name The key name
15110          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15111          * @return {Mixed} The state data
15112          */
15113         get : function(key, defaultValue){
15114             return provider.get(key, defaultValue);
15115         },
15116         
15117         /**
15118          * Sets the value for a key
15119          * @param {String} name The key name
15120          * @param {Mixed} value The state data
15121          */
15122          set : function(key, value){
15123             provider.set(key, value);
15124         },
15125         
15126         /**
15127          * Clears a value from the state
15128          * @param {String} name The key name
15129          */
15130         clear : function(key){
15131             provider.clear(key);
15132         },
15133         
15134         /**
15135          * Gets the currently configured state provider
15136          * @return {Provider} The state provider
15137          */
15138         getProvider : function(){
15139             return provider;
15140         }
15141     };
15142 }();
15143 /*
15144  * Based on:
15145  * Ext JS Library 1.1.1
15146  * Copyright(c) 2006-2007, Ext JS, LLC.
15147  *
15148  * Originally Released Under LGPL - original licence link has changed is not relivant.
15149  *
15150  * Fork - LGPL
15151  * <script type="text/javascript">
15152  */
15153 /**
15154  * @class Roo.state.CookieProvider
15155  * @extends Roo.state.Provider
15156  * The default Provider implementation which saves state via cookies.
15157  * <br />Usage:
15158  <pre><code>
15159    var cp = new Roo.state.CookieProvider({
15160        path: "/cgi-bin/",
15161        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15162        domain: "roojs.com"
15163    })
15164    Roo.state.Manager.setProvider(cp);
15165  </code></pre>
15166  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15167  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15168  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15169  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15170  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15171  * domain the page is running on including the 'www' like 'www.roojs.com')
15172  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15173  * @constructor
15174  * Create a new CookieProvider
15175  * @param {Object} config The configuration object
15176  */
15177 Roo.state.CookieProvider = function(config){
15178     Roo.state.CookieProvider.superclass.constructor.call(this);
15179     this.path = "/";
15180     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15181     this.domain = null;
15182     this.secure = false;
15183     Roo.apply(this, config);
15184     this.state = this.readCookies();
15185 };
15186
15187 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15188     // private
15189     set : function(name, value){
15190         if(typeof value == "undefined" || value === null){
15191             this.clear(name);
15192             return;
15193         }
15194         this.setCookie(name, value);
15195         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15196     },
15197
15198     // private
15199     clear : function(name){
15200         this.clearCookie(name);
15201         Roo.state.CookieProvider.superclass.clear.call(this, name);
15202     },
15203
15204     // private
15205     readCookies : function(){
15206         var cookies = {};
15207         var c = document.cookie + ";";
15208         var re = /\s?(.*?)=(.*?);/g;
15209         var matches;
15210         while((matches = re.exec(c)) != null){
15211             var name = matches[1];
15212             var value = matches[2];
15213             if(name && name.substring(0,3) == "ys-"){
15214                 cookies[name.substr(3)] = this.decodeValue(value);
15215             }
15216         }
15217         return cookies;
15218     },
15219
15220     // private
15221     setCookie : function(name, value){
15222         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15223            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15224            ((this.path == null) ? "" : ("; path=" + this.path)) +
15225            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15226            ((this.secure == true) ? "; secure" : "");
15227     },
15228
15229     // private
15230     clearCookie : function(name){
15231         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15232            ((this.path == null) ? "" : ("; path=" + this.path)) +
15233            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15234            ((this.secure == true) ? "; secure" : "");
15235     }
15236 });/*
15237  * Based on:
15238  * Ext JS Library 1.1.1
15239  * Copyright(c) 2006-2007, Ext JS, LLC.
15240  *
15241  * Originally Released Under LGPL - original licence link has changed is not relivant.
15242  *
15243  * Fork - LGPL
15244  * <script type="text/javascript">
15245  */
15246  
15247
15248 /**
15249  * @class Roo.ComponentMgr
15250  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15251  * @singleton
15252  */
15253 Roo.ComponentMgr = function(){
15254     var all = new Roo.util.MixedCollection();
15255
15256     return {
15257         /**
15258          * Registers a component.
15259          * @param {Roo.Component} c The component
15260          */
15261         register : function(c){
15262             all.add(c);
15263         },
15264
15265         /**
15266          * Unregisters a component.
15267          * @param {Roo.Component} c The component
15268          */
15269         unregister : function(c){
15270             all.remove(c);
15271         },
15272
15273         /**
15274          * Returns a component by id
15275          * @param {String} id The component id
15276          */
15277         get : function(id){
15278             return all.get(id);
15279         },
15280
15281         /**
15282          * Registers a function that will be called when a specified component is added to ComponentMgr
15283          * @param {String} id The component id
15284          * @param {Funtction} fn The callback function
15285          * @param {Object} scope The scope of the callback
15286          */
15287         onAvailable : function(id, fn, scope){
15288             all.on("add", function(index, o){
15289                 if(o.id == id){
15290                     fn.call(scope || o, o);
15291                     all.un("add", fn, scope);
15292                 }
15293             });
15294         }
15295     };
15296 }();/*
15297  * Based on:
15298  * Ext JS Library 1.1.1
15299  * Copyright(c) 2006-2007, Ext JS, LLC.
15300  *
15301  * Originally Released Under LGPL - original licence link has changed is not relivant.
15302  *
15303  * Fork - LGPL
15304  * <script type="text/javascript">
15305  */
15306  
15307 /**
15308  * @class Roo.Component
15309  * @extends Roo.util.Observable
15310  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15311  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15312  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15313  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15314  * All visual components (widgets) that require rendering into a layout should subclass Component.
15315  * @constructor
15316  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15317  * 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
15318  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15319  */
15320 Roo.Component = function(config){
15321     config = config || {};
15322     if(config.tagName || config.dom || typeof config == "string"){ // element object
15323         config = {el: config, id: config.id || config};
15324     }
15325     this.initialConfig = config;
15326
15327     Roo.apply(this, config);
15328     this.addEvents({
15329         /**
15330          * @event disable
15331          * Fires after the component is disabled.
15332              * @param {Roo.Component} this
15333              */
15334         disable : true,
15335         /**
15336          * @event enable
15337          * Fires after the component is enabled.
15338              * @param {Roo.Component} this
15339              */
15340         enable : true,
15341         /**
15342          * @event beforeshow
15343          * Fires before the component is shown.  Return false to stop the show.
15344              * @param {Roo.Component} this
15345              */
15346         beforeshow : true,
15347         /**
15348          * @event show
15349          * Fires after the component is shown.
15350              * @param {Roo.Component} this
15351              */
15352         show : true,
15353         /**
15354          * @event beforehide
15355          * Fires before the component is hidden. Return false to stop the hide.
15356              * @param {Roo.Component} this
15357              */
15358         beforehide : true,
15359         /**
15360          * @event hide
15361          * Fires after the component is hidden.
15362              * @param {Roo.Component} this
15363              */
15364         hide : true,
15365         /**
15366          * @event beforerender
15367          * Fires before the component is rendered. Return false to stop the render.
15368              * @param {Roo.Component} this
15369              */
15370         beforerender : true,
15371         /**
15372          * @event render
15373          * Fires after the component is rendered.
15374              * @param {Roo.Component} this
15375              */
15376         render : true,
15377         /**
15378          * @event beforedestroy
15379          * Fires before the component is destroyed. Return false to stop the destroy.
15380              * @param {Roo.Component} this
15381              */
15382         beforedestroy : true,
15383         /**
15384          * @event destroy
15385          * Fires after the component is destroyed.
15386              * @param {Roo.Component} this
15387              */
15388         destroy : true
15389     });
15390     if(!this.id){
15391         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15392     }
15393     Roo.ComponentMgr.register(this);
15394     Roo.Component.superclass.constructor.call(this);
15395     this.initComponent();
15396     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15397         this.render(this.renderTo);
15398         delete this.renderTo;
15399     }
15400 };
15401
15402 /** @private */
15403 Roo.Component.AUTO_ID = 1000;
15404
15405 Roo.extend(Roo.Component, Roo.util.Observable, {
15406     /**
15407      * @scope Roo.Component.prototype
15408      * @type {Boolean}
15409      * true if this component is hidden. Read-only.
15410      */
15411     hidden : false,
15412     /**
15413      * @type {Boolean}
15414      * true if this component is disabled. Read-only.
15415      */
15416     disabled : false,
15417     /**
15418      * @type {Boolean}
15419      * true if this component has been rendered. Read-only.
15420      */
15421     rendered : false,
15422     
15423     /** @cfg {String} disableClass
15424      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15425      */
15426     disabledClass : "x-item-disabled",
15427         /** @cfg {Boolean} allowDomMove
15428          * Whether the component can move the Dom node when rendering (defaults to true).
15429          */
15430     allowDomMove : true,
15431     /** @cfg {String} hideMode (display|visibility)
15432      * How this component should hidden. Supported values are
15433      * "visibility" (css visibility), "offsets" (negative offset position) and
15434      * "display" (css display) - defaults to "display".
15435      */
15436     hideMode: 'display',
15437
15438     /** @private */
15439     ctype : "Roo.Component",
15440
15441     /**
15442      * @cfg {String} actionMode 
15443      * which property holds the element that used for  hide() / show() / disable() / enable()
15444      * default is 'el' for forms you probably want to set this to fieldEl 
15445      */
15446     actionMode : "el",
15447
15448     /** @private */
15449     getActionEl : function(){
15450         return this[this.actionMode];
15451     },
15452
15453     initComponent : Roo.emptyFn,
15454     /**
15455      * If this is a lazy rendering component, render it to its container element.
15456      * @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.
15457      */
15458     render : function(container, position){
15459         
15460         if(this.rendered){
15461             return this;
15462         }
15463         
15464         if(this.fireEvent("beforerender", this) === false){
15465             return false;
15466         }
15467         
15468         if(!container && this.el){
15469             this.el = Roo.get(this.el);
15470             container = this.el.dom.parentNode;
15471             this.allowDomMove = false;
15472         }
15473         this.container = Roo.get(container);
15474         this.rendered = true;
15475         if(position !== undefined){
15476             if(typeof position == 'number'){
15477                 position = this.container.dom.childNodes[position];
15478             }else{
15479                 position = Roo.getDom(position);
15480             }
15481         }
15482         this.onRender(this.container, position || null);
15483         if(this.cls){
15484             this.el.addClass(this.cls);
15485             delete this.cls;
15486         }
15487         if(this.style){
15488             this.el.applyStyles(this.style);
15489             delete this.style;
15490         }
15491         this.fireEvent("render", this);
15492         this.afterRender(this.container);
15493         if(this.hidden){
15494             this.hide();
15495         }
15496         if(this.disabled){
15497             this.disable();
15498         }
15499
15500         return this;
15501         
15502     },
15503
15504     /** @private */
15505     // default function is not really useful
15506     onRender : function(ct, position){
15507         if(this.el){
15508             this.el = Roo.get(this.el);
15509             if(this.allowDomMove !== false){
15510                 ct.dom.insertBefore(this.el.dom, position);
15511             }
15512         }
15513     },
15514
15515     /** @private */
15516     getAutoCreate : function(){
15517         var cfg = typeof this.autoCreate == "object" ?
15518                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15519         if(this.id && !cfg.id){
15520             cfg.id = this.id;
15521         }
15522         return cfg;
15523     },
15524
15525     /** @private */
15526     afterRender : Roo.emptyFn,
15527
15528     /**
15529      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15530      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15531      */
15532     destroy : function(){
15533         if(this.fireEvent("beforedestroy", this) !== false){
15534             this.purgeListeners();
15535             this.beforeDestroy();
15536             if(this.rendered){
15537                 this.el.removeAllListeners();
15538                 this.el.remove();
15539                 if(this.actionMode == "container"){
15540                     this.container.remove();
15541                 }
15542             }
15543             this.onDestroy();
15544             Roo.ComponentMgr.unregister(this);
15545             this.fireEvent("destroy", this);
15546         }
15547     },
15548
15549         /** @private */
15550     beforeDestroy : function(){
15551
15552     },
15553
15554         /** @private */
15555         onDestroy : function(){
15556
15557     },
15558
15559     /**
15560      * Returns the underlying {@link Roo.Element}.
15561      * @return {Roo.Element} The element
15562      */
15563     getEl : function(){
15564         return this.el;
15565     },
15566
15567     /**
15568      * Returns the id of this component.
15569      * @return {String}
15570      */
15571     getId : function(){
15572         return this.id;
15573     },
15574
15575     /**
15576      * Try to focus this component.
15577      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15578      * @return {Roo.Component} this
15579      */
15580     focus : function(selectText){
15581         if(this.rendered){
15582             this.el.focus();
15583             if(selectText === true){
15584                 this.el.dom.select();
15585             }
15586         }
15587         return this;
15588     },
15589
15590     /** @private */
15591     blur : function(){
15592         if(this.rendered){
15593             this.el.blur();
15594         }
15595         return this;
15596     },
15597
15598     /**
15599      * Disable this component.
15600      * @return {Roo.Component} this
15601      */
15602     disable : function(){
15603         if(this.rendered){
15604             this.onDisable();
15605         }
15606         this.disabled = true;
15607         this.fireEvent("disable", this);
15608         return this;
15609     },
15610
15611         // private
15612     onDisable : function(){
15613         this.getActionEl().addClass(this.disabledClass);
15614         this.el.dom.disabled = true;
15615     },
15616
15617     /**
15618      * Enable this component.
15619      * @return {Roo.Component} this
15620      */
15621     enable : function(){
15622         if(this.rendered){
15623             this.onEnable();
15624         }
15625         this.disabled = false;
15626         this.fireEvent("enable", this);
15627         return this;
15628     },
15629
15630         // private
15631     onEnable : function(){
15632         this.getActionEl().removeClass(this.disabledClass);
15633         this.el.dom.disabled = false;
15634     },
15635
15636     /**
15637      * Convenience function for setting disabled/enabled by boolean.
15638      * @param {Boolean} disabled
15639      */
15640     setDisabled : function(disabled){
15641         this[disabled ? "disable" : "enable"]();
15642     },
15643
15644     /**
15645      * Show this component.
15646      * @return {Roo.Component} this
15647      */
15648     show: function(){
15649         if(this.fireEvent("beforeshow", this) !== false){
15650             this.hidden = false;
15651             if(this.rendered){
15652                 this.onShow();
15653             }
15654             this.fireEvent("show", this);
15655         }
15656         return this;
15657     },
15658
15659     // private
15660     onShow : function(){
15661         var ae = this.getActionEl();
15662         if(this.hideMode == 'visibility'){
15663             ae.dom.style.visibility = "visible";
15664         }else if(this.hideMode == 'offsets'){
15665             ae.removeClass('x-hidden');
15666         }else{
15667             ae.dom.style.display = "";
15668         }
15669     },
15670
15671     /**
15672      * Hide this component.
15673      * @return {Roo.Component} this
15674      */
15675     hide: function(){
15676         if(this.fireEvent("beforehide", this) !== false){
15677             this.hidden = true;
15678             if(this.rendered){
15679                 this.onHide();
15680             }
15681             this.fireEvent("hide", this);
15682         }
15683         return this;
15684     },
15685
15686     // private
15687     onHide : function(){
15688         var ae = this.getActionEl();
15689         if(this.hideMode == 'visibility'){
15690             ae.dom.style.visibility = "hidden";
15691         }else if(this.hideMode == 'offsets'){
15692             ae.addClass('x-hidden');
15693         }else{
15694             ae.dom.style.display = "none";
15695         }
15696     },
15697
15698     /**
15699      * Convenience function to hide or show this component by boolean.
15700      * @param {Boolean} visible True to show, false to hide
15701      * @return {Roo.Component} this
15702      */
15703     setVisible: function(visible){
15704         if(visible) {
15705             this.show();
15706         }else{
15707             this.hide();
15708         }
15709         return this;
15710     },
15711
15712     /**
15713      * Returns true if this component is visible.
15714      */
15715     isVisible : function(){
15716         return this.getActionEl().isVisible();
15717     },
15718
15719     cloneConfig : function(overrides){
15720         overrides = overrides || {};
15721         var id = overrides.id || Roo.id();
15722         var cfg = Roo.applyIf(overrides, this.initialConfig);
15723         cfg.id = id; // prevent dup id
15724         return new this.constructor(cfg);
15725     }
15726 });/*
15727  * Based on:
15728  * Ext JS Library 1.1.1
15729  * Copyright(c) 2006-2007, Ext JS, LLC.
15730  *
15731  * Originally Released Under LGPL - original licence link has changed is not relivant.
15732  *
15733  * Fork - LGPL
15734  * <script type="text/javascript">
15735  */
15736
15737 /**
15738  * @class Roo.BoxComponent
15739  * @extends Roo.Component
15740  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15741  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15742  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15743  * layout containers.
15744  * @constructor
15745  * @param {Roo.Element/String/Object} config The configuration options.
15746  */
15747 Roo.BoxComponent = function(config){
15748     Roo.Component.call(this, config);
15749     this.addEvents({
15750         /**
15751          * @event resize
15752          * Fires after the component is resized.
15753              * @param {Roo.Component} this
15754              * @param {Number} adjWidth The box-adjusted width that was set
15755              * @param {Number} adjHeight The box-adjusted height that was set
15756              * @param {Number} rawWidth The width that was originally specified
15757              * @param {Number} rawHeight The height that was originally specified
15758              */
15759         resize : true,
15760         /**
15761          * @event move
15762          * Fires after the component is moved.
15763              * @param {Roo.Component} this
15764              * @param {Number} x The new x position
15765              * @param {Number} y The new y position
15766              */
15767         move : true
15768     });
15769 };
15770
15771 Roo.extend(Roo.BoxComponent, Roo.Component, {
15772     // private, set in afterRender to signify that the component has been rendered
15773     boxReady : false,
15774     // private, used to defer height settings to subclasses
15775     deferHeight: false,
15776     /** @cfg {Number} width
15777      * width (optional) size of component
15778      */
15779      /** @cfg {Number} height
15780      * height (optional) size of component
15781      */
15782      
15783     /**
15784      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15785      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15786      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15787      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15788      * @return {Roo.BoxComponent} this
15789      */
15790     setSize : function(w, h){
15791         // support for standard size objects
15792         if(typeof w == 'object'){
15793             h = w.height;
15794             w = w.width;
15795         }
15796         // not rendered
15797         if(!this.boxReady){
15798             this.width = w;
15799             this.height = h;
15800             return this;
15801         }
15802
15803         // prevent recalcs when not needed
15804         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15805             return this;
15806         }
15807         this.lastSize = {width: w, height: h};
15808
15809         var adj = this.adjustSize(w, h);
15810         var aw = adj.width, ah = adj.height;
15811         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15812             var rz = this.getResizeEl();
15813             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15814                 rz.setSize(aw, ah);
15815             }else if(!this.deferHeight && ah !== undefined){
15816                 rz.setHeight(ah);
15817             }else if(aw !== undefined){
15818                 rz.setWidth(aw);
15819             }
15820             this.onResize(aw, ah, w, h);
15821             this.fireEvent('resize', this, aw, ah, w, h);
15822         }
15823         return this;
15824     },
15825
15826     /**
15827      * Gets the current size of the component's underlying element.
15828      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15829      */
15830     getSize : function(){
15831         return this.el.getSize();
15832     },
15833
15834     /**
15835      * Gets the current XY position of the component's underlying element.
15836      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15837      * @return {Array} The XY position of the element (e.g., [100, 200])
15838      */
15839     getPosition : function(local){
15840         if(local === true){
15841             return [this.el.getLeft(true), this.el.getTop(true)];
15842         }
15843         return this.xy || this.el.getXY();
15844     },
15845
15846     /**
15847      * Gets the current box measurements of the component's underlying element.
15848      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15849      * @returns {Object} box An object in the format {x, y, width, height}
15850      */
15851     getBox : function(local){
15852         var s = this.el.getSize();
15853         if(local){
15854             s.x = this.el.getLeft(true);
15855             s.y = this.el.getTop(true);
15856         }else{
15857             var xy = this.xy || this.el.getXY();
15858             s.x = xy[0];
15859             s.y = xy[1];
15860         }
15861         return s;
15862     },
15863
15864     /**
15865      * Sets the current box measurements of the component's underlying element.
15866      * @param {Object} box An object in the format {x, y, width, height}
15867      * @returns {Roo.BoxComponent} this
15868      */
15869     updateBox : function(box){
15870         this.setSize(box.width, box.height);
15871         this.setPagePosition(box.x, box.y);
15872         return this;
15873     },
15874
15875     // protected
15876     getResizeEl : function(){
15877         return this.resizeEl || this.el;
15878     },
15879
15880     // protected
15881     getPositionEl : function(){
15882         return this.positionEl || this.el;
15883     },
15884
15885     /**
15886      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15887      * This method fires the move event.
15888      * @param {Number} left The new left
15889      * @param {Number} top The new top
15890      * @returns {Roo.BoxComponent} this
15891      */
15892     setPosition : function(x, y){
15893         this.x = x;
15894         this.y = y;
15895         if(!this.boxReady){
15896             return this;
15897         }
15898         var adj = this.adjustPosition(x, y);
15899         var ax = adj.x, ay = adj.y;
15900
15901         var el = this.getPositionEl();
15902         if(ax !== undefined || ay !== undefined){
15903             if(ax !== undefined && ay !== undefined){
15904                 el.setLeftTop(ax, ay);
15905             }else if(ax !== undefined){
15906                 el.setLeft(ax);
15907             }else if(ay !== undefined){
15908                 el.setTop(ay);
15909             }
15910             this.onPosition(ax, ay);
15911             this.fireEvent('move', this, ax, ay);
15912         }
15913         return this;
15914     },
15915
15916     /**
15917      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15918      * This method fires the move event.
15919      * @param {Number} x The new x position
15920      * @param {Number} y The new y position
15921      * @returns {Roo.BoxComponent} this
15922      */
15923     setPagePosition : function(x, y){
15924         this.pageX = x;
15925         this.pageY = y;
15926         if(!this.boxReady){
15927             return;
15928         }
15929         if(x === undefined || y === undefined){ // cannot translate undefined points
15930             return;
15931         }
15932         var p = this.el.translatePoints(x, y);
15933         this.setPosition(p.left, p.top);
15934         return this;
15935     },
15936
15937     // private
15938     onRender : function(ct, position){
15939         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15940         if(this.resizeEl){
15941             this.resizeEl = Roo.get(this.resizeEl);
15942         }
15943         if(this.positionEl){
15944             this.positionEl = Roo.get(this.positionEl);
15945         }
15946     },
15947
15948     // private
15949     afterRender : function(){
15950         Roo.BoxComponent.superclass.afterRender.call(this);
15951         this.boxReady = true;
15952         this.setSize(this.width, this.height);
15953         if(this.x || this.y){
15954             this.setPosition(this.x, this.y);
15955         }
15956         if(this.pageX || this.pageY){
15957             this.setPagePosition(this.pageX, this.pageY);
15958         }
15959     },
15960
15961     /**
15962      * Force the component's size to recalculate based on the underlying element's current height and width.
15963      * @returns {Roo.BoxComponent} this
15964      */
15965     syncSize : function(){
15966         delete this.lastSize;
15967         this.setSize(this.el.getWidth(), this.el.getHeight());
15968         return this;
15969     },
15970
15971     /**
15972      * Called after the component is resized, this method is empty by default but can be implemented by any
15973      * subclass that needs to perform custom logic after a resize occurs.
15974      * @param {Number} adjWidth The box-adjusted width that was set
15975      * @param {Number} adjHeight The box-adjusted height that was set
15976      * @param {Number} rawWidth The width that was originally specified
15977      * @param {Number} rawHeight The height that was originally specified
15978      */
15979     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15980
15981     },
15982
15983     /**
15984      * Called after the component is moved, this method is empty by default but can be implemented by any
15985      * subclass that needs to perform custom logic after a move occurs.
15986      * @param {Number} x The new x position
15987      * @param {Number} y The new y position
15988      */
15989     onPosition : function(x, y){
15990
15991     },
15992
15993     // private
15994     adjustSize : function(w, h){
15995         if(this.autoWidth){
15996             w = 'auto';
15997         }
15998         if(this.autoHeight){
15999             h = 'auto';
16000         }
16001         return {width : w, height: h};
16002     },
16003
16004     // private
16005     adjustPosition : function(x, y){
16006         return {x : x, y: y};
16007     }
16008 });/*
16009  * Based on:
16010  * Ext JS Library 1.1.1
16011  * Copyright(c) 2006-2007, Ext JS, LLC.
16012  *
16013  * Originally Released Under LGPL - original licence link has changed is not relivant.
16014  *
16015  * Fork - LGPL
16016  * <script type="text/javascript">
16017  */
16018  (function(){ 
16019 /**
16020  * @class Roo.Layer
16021  * @extends Roo.Element
16022  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
16023  * automatic maintaining of shadow/shim positions.
16024  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
16025  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
16026  * you can pass a string with a CSS class name. False turns off the shadow.
16027  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
16028  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
16029  * @cfg {String} cls CSS class to add to the element
16030  * @cfg {Number} zindex Starting z-index (defaults to 11000)
16031  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
16032  * @constructor
16033  * @param {Object} config An object with config options.
16034  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
16035  */
16036
16037 Roo.Layer = function(config, existingEl){
16038     config = config || {};
16039     var dh = Roo.DomHelper;
16040     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
16041     if(existingEl){
16042         this.dom = Roo.getDom(existingEl);
16043     }
16044     if(!this.dom){
16045         var o = config.dh || {tag: "div", cls: "x-layer"};
16046         this.dom = dh.append(pel, o);
16047     }
16048     if(config.cls){
16049         this.addClass(config.cls);
16050     }
16051     this.constrain = config.constrain !== false;
16052     this.visibilityMode = Roo.Element.VISIBILITY;
16053     if(config.id){
16054         this.id = this.dom.id = config.id;
16055     }else{
16056         this.id = Roo.id(this.dom);
16057     }
16058     this.zindex = config.zindex || this.getZIndex();
16059     this.position("absolute", this.zindex);
16060     if(config.shadow){
16061         this.shadowOffset = config.shadowOffset || 4;
16062         this.shadow = new Roo.Shadow({
16063             offset : this.shadowOffset,
16064             mode : config.shadow
16065         });
16066     }else{
16067         this.shadowOffset = 0;
16068     }
16069     this.useShim = config.shim !== false && Roo.useShims;
16070     this.useDisplay = config.useDisplay;
16071     this.hide();
16072 };
16073
16074 var supr = Roo.Element.prototype;
16075
16076 // shims are shared among layer to keep from having 100 iframes
16077 var shims = [];
16078
16079 Roo.extend(Roo.Layer, Roo.Element, {
16080
16081     getZIndex : function(){
16082         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
16083     },
16084
16085     getShim : function(){
16086         if(!this.useShim){
16087             return null;
16088         }
16089         if(this.shim){
16090             return this.shim;
16091         }
16092         var shim = shims.shift();
16093         if(!shim){
16094             shim = this.createShim();
16095             shim.enableDisplayMode('block');
16096             shim.dom.style.display = 'none';
16097             shim.dom.style.visibility = 'visible';
16098         }
16099         var pn = this.dom.parentNode;
16100         if(shim.dom.parentNode != pn){
16101             pn.insertBefore(shim.dom, this.dom);
16102         }
16103         shim.setStyle('z-index', this.getZIndex()-2);
16104         this.shim = shim;
16105         return shim;
16106     },
16107
16108     hideShim : function(){
16109         if(this.shim){
16110             this.shim.setDisplayed(false);
16111             shims.push(this.shim);
16112             delete this.shim;
16113         }
16114     },
16115
16116     disableShadow : function(){
16117         if(this.shadow){
16118             this.shadowDisabled = true;
16119             this.shadow.hide();
16120             this.lastShadowOffset = this.shadowOffset;
16121             this.shadowOffset = 0;
16122         }
16123     },
16124
16125     enableShadow : function(show){
16126         if(this.shadow){
16127             this.shadowDisabled = false;
16128             this.shadowOffset = this.lastShadowOffset;
16129             delete this.lastShadowOffset;
16130             if(show){
16131                 this.sync(true);
16132             }
16133         }
16134     },
16135
16136     // private
16137     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
16138     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
16139     sync : function(doShow){
16140         var sw = this.shadow;
16141         if(!this.updating && this.isVisible() && (sw || this.useShim)){
16142             var sh = this.getShim();
16143
16144             var w = this.getWidth(),
16145                 h = this.getHeight();
16146
16147             var l = this.getLeft(true),
16148                 t = this.getTop(true);
16149
16150             if(sw && !this.shadowDisabled){
16151                 if(doShow && !sw.isVisible()){
16152                     sw.show(this);
16153                 }else{
16154                     sw.realign(l, t, w, h);
16155                 }
16156                 if(sh){
16157                     if(doShow){
16158                        sh.show();
16159                     }
16160                     // fit the shim behind the shadow, so it is shimmed too
16161                     var a = sw.adjusts, s = sh.dom.style;
16162                     s.left = (Math.min(l, l+a.l))+"px";
16163                     s.top = (Math.min(t, t+a.t))+"px";
16164                     s.width = (w+a.w)+"px";
16165                     s.height = (h+a.h)+"px";
16166                 }
16167             }else if(sh){
16168                 if(doShow){
16169                    sh.show();
16170                 }
16171                 sh.setSize(w, h);
16172                 sh.setLeftTop(l, t);
16173             }
16174             
16175         }
16176     },
16177
16178     // private
16179     destroy : function(){
16180         this.hideShim();
16181         if(this.shadow){
16182             this.shadow.hide();
16183         }
16184         this.removeAllListeners();
16185         var pn = this.dom.parentNode;
16186         if(pn){
16187             pn.removeChild(this.dom);
16188         }
16189         Roo.Element.uncache(this.id);
16190     },
16191
16192     remove : function(){
16193         this.destroy();
16194     },
16195
16196     // private
16197     beginUpdate : function(){
16198         this.updating = true;
16199     },
16200
16201     // private
16202     endUpdate : function(){
16203         this.updating = false;
16204         this.sync(true);
16205     },
16206
16207     // private
16208     hideUnders : function(negOffset){
16209         if(this.shadow){
16210             this.shadow.hide();
16211         }
16212         this.hideShim();
16213     },
16214
16215     // private
16216     constrainXY : function(){
16217         if(this.constrain){
16218             var vw = Roo.lib.Dom.getViewWidth(),
16219                 vh = Roo.lib.Dom.getViewHeight();
16220             var s = Roo.get(document).getScroll();
16221
16222             var xy = this.getXY();
16223             var x = xy[0], y = xy[1];   
16224             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
16225             // only move it if it needs it
16226             var moved = false;
16227             // first validate right/bottom
16228             if((x + w) > vw+s.left){
16229                 x = vw - w - this.shadowOffset;
16230                 moved = true;
16231             }
16232             if((y + h) > vh+s.top){
16233                 y = vh - h - this.shadowOffset;
16234                 moved = true;
16235             }
16236             // then make sure top/left isn't negative
16237             if(x < s.left){
16238                 x = s.left;
16239                 moved = true;
16240             }
16241             if(y < s.top){
16242                 y = s.top;
16243                 moved = true;
16244             }
16245             if(moved){
16246                 if(this.avoidY){
16247                     var ay = this.avoidY;
16248                     if(y <= ay && (y+h) >= ay){
16249                         y = ay-h-5;   
16250                     }
16251                 }
16252                 xy = [x, y];
16253                 this.storeXY(xy);
16254                 supr.setXY.call(this, xy);
16255                 this.sync();
16256             }
16257         }
16258     },
16259
16260     isVisible : function(){
16261         return this.visible;    
16262     },
16263
16264     // private
16265     showAction : function(){
16266         this.visible = true; // track visibility to prevent getStyle calls
16267         if(this.useDisplay === true){
16268             this.setDisplayed("");
16269         }else if(this.lastXY){
16270             supr.setXY.call(this, this.lastXY);
16271         }else if(this.lastLT){
16272             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
16273         }
16274     },
16275
16276     // private
16277     hideAction : function(){
16278         this.visible = false;
16279         if(this.useDisplay === true){
16280             this.setDisplayed(false);
16281         }else{
16282             this.setLeftTop(-10000,-10000);
16283         }
16284     },
16285
16286     // overridden Element method
16287     setVisible : function(v, a, d, c, e){
16288         if(v){
16289             this.showAction();
16290         }
16291         if(a && v){
16292             var cb = function(){
16293                 this.sync(true);
16294                 if(c){
16295                     c();
16296                 }
16297             }.createDelegate(this);
16298             supr.setVisible.call(this, true, true, d, cb, e);
16299         }else{
16300             if(!v){
16301                 this.hideUnders(true);
16302             }
16303             var cb = c;
16304             if(a){
16305                 cb = function(){
16306                     this.hideAction();
16307                     if(c){
16308                         c();
16309                     }
16310                 }.createDelegate(this);
16311             }
16312             supr.setVisible.call(this, v, a, d, cb, e);
16313             if(v){
16314                 this.sync(true);
16315             }else if(!a){
16316                 this.hideAction();
16317             }
16318         }
16319     },
16320
16321     storeXY : function(xy){
16322         delete this.lastLT;
16323         this.lastXY = xy;
16324     },
16325
16326     storeLeftTop : function(left, top){
16327         delete this.lastXY;
16328         this.lastLT = [left, top];
16329     },
16330
16331     // private
16332     beforeFx : function(){
16333         this.beforeAction();
16334         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
16335     },
16336
16337     // private
16338     afterFx : function(){
16339         Roo.Layer.superclass.afterFx.apply(this, arguments);
16340         this.sync(this.isVisible());
16341     },
16342
16343     // private
16344     beforeAction : function(){
16345         if(!this.updating && this.shadow){
16346             this.shadow.hide();
16347         }
16348     },
16349
16350     // overridden Element method
16351     setLeft : function(left){
16352         this.storeLeftTop(left, this.getTop(true));
16353         supr.setLeft.apply(this, arguments);
16354         this.sync();
16355     },
16356
16357     setTop : function(top){
16358         this.storeLeftTop(this.getLeft(true), top);
16359         supr.setTop.apply(this, arguments);
16360         this.sync();
16361     },
16362
16363     setLeftTop : function(left, top){
16364         this.storeLeftTop(left, top);
16365         supr.setLeftTop.apply(this, arguments);
16366         this.sync();
16367     },
16368
16369     setXY : function(xy, a, d, c, e){
16370         this.fixDisplay();
16371         this.beforeAction();
16372         this.storeXY(xy);
16373         var cb = this.createCB(c);
16374         supr.setXY.call(this, xy, a, d, cb, e);
16375         if(!a){
16376             cb();
16377         }
16378     },
16379
16380     // private
16381     createCB : function(c){
16382         var el = this;
16383         return function(){
16384             el.constrainXY();
16385             el.sync(true);
16386             if(c){
16387                 c();
16388             }
16389         };
16390     },
16391
16392     // overridden Element method
16393     setX : function(x, a, d, c, e){
16394         this.setXY([x, this.getY()], a, d, c, e);
16395     },
16396
16397     // overridden Element method
16398     setY : function(y, a, d, c, e){
16399         this.setXY([this.getX(), y], a, d, c, e);
16400     },
16401
16402     // overridden Element method
16403     setSize : function(w, h, a, d, c, e){
16404         this.beforeAction();
16405         var cb = this.createCB(c);
16406         supr.setSize.call(this, w, h, a, d, cb, e);
16407         if(!a){
16408             cb();
16409         }
16410     },
16411
16412     // overridden Element method
16413     setWidth : function(w, a, d, c, e){
16414         this.beforeAction();
16415         var cb = this.createCB(c);
16416         supr.setWidth.call(this, w, a, d, cb, e);
16417         if(!a){
16418             cb();
16419         }
16420     },
16421
16422     // overridden Element method
16423     setHeight : function(h, a, d, c, e){
16424         this.beforeAction();
16425         var cb = this.createCB(c);
16426         supr.setHeight.call(this, h, a, d, cb, e);
16427         if(!a){
16428             cb();
16429         }
16430     },
16431
16432     // overridden Element method
16433     setBounds : function(x, y, w, h, a, d, c, e){
16434         this.beforeAction();
16435         var cb = this.createCB(c);
16436         if(!a){
16437             this.storeXY([x, y]);
16438             supr.setXY.call(this, [x, y]);
16439             supr.setSize.call(this, w, h, a, d, cb, e);
16440             cb();
16441         }else{
16442             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
16443         }
16444         return this;
16445     },
16446     
16447     /**
16448      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
16449      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
16450      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
16451      * @param {Number} zindex The new z-index to set
16452      * @return {this} The Layer
16453      */
16454     setZIndex : function(zindex){
16455         this.zindex = zindex;
16456         this.setStyle("z-index", zindex + 2);
16457         if(this.shadow){
16458             this.shadow.setZIndex(zindex + 1);
16459         }
16460         if(this.shim){
16461             this.shim.setStyle("z-index", zindex);
16462         }
16463     }
16464 });
16465 })();/*
16466  * Original code for Roojs - LGPL
16467  * <script type="text/javascript">
16468  */
16469  
16470 /**
16471  * @class Roo.XComponent
16472  * A delayed Element creator...
16473  * Or a way to group chunks of interface together.
16474  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16475  *  used in conjunction with XComponent.build() it will create an instance of each element,
16476  *  then call addxtype() to build the User interface.
16477  * 
16478  * Mypart.xyx = new Roo.XComponent({
16479
16480     parent : 'Mypart.xyz', // empty == document.element.!!
16481     order : '001',
16482     name : 'xxxx'
16483     region : 'xxxx'
16484     disabled : function() {} 
16485      
16486     tree : function() { // return an tree of xtype declared components
16487         var MODULE = this;
16488         return 
16489         {
16490             xtype : 'NestedLayoutPanel',
16491             // technicall
16492         }
16493      ]
16494  *})
16495  *
16496  *
16497  * It can be used to build a big heiracy, with parent etc.
16498  * or you can just use this to render a single compoent to a dom element
16499  * MYPART.render(Roo.Element | String(id) | dom_element )
16500  *
16501  *
16502  * Usage patterns.
16503  *
16504  * Classic Roo
16505  *
16506  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16507  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16508  *
16509  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16510  *
16511  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16512  * - if mulitple topModules exist, the last one is defined as the top module.
16513  *
16514  * Embeded Roo
16515  * 
16516  * When the top level or multiple modules are to embedded into a existing HTML page,
16517  * the parent element can container '#id' of the element where the module will be drawn.
16518  *
16519  * Bootstrap Roo
16520  *
16521  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16522  * it relies more on a include mechanism, where sub modules are included into an outer page.
16523  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16524  * 
16525  * Bootstrap Roo Included elements
16526  *
16527  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16528  * hence confusing the component builder as it thinks there are multiple top level elements. 
16529  *
16530  * String Over-ride & Translations
16531  *
16532  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16533  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16534  * are needed. @see Roo.XComponent.overlayString  
16535  * 
16536  * 
16537  * 
16538  * @extends Roo.util.Observable
16539  * @constructor
16540  * @param cfg {Object} configuration of component
16541  * 
16542  */
16543 Roo.XComponent = function(cfg) {
16544     Roo.apply(this, cfg);
16545     this.addEvents({ 
16546         /**
16547              * @event built
16548              * Fires when this the componnt is built
16549              * @param {Roo.XComponent} c the component
16550              */
16551         'built' : true
16552         
16553     });
16554     this.region = this.region || 'center'; // default..
16555     Roo.XComponent.register(this);
16556     this.modules = false;
16557     this.el = false; // where the layout goes..
16558     
16559     
16560 }
16561 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16562     /**
16563      * @property el
16564      * The created element (with Roo.factory())
16565      * @type {Roo.Layout}
16566      */
16567     el  : false,
16568     
16569     /**
16570      * @property el
16571      * for BC  - use el in new code
16572      * @type {Roo.Layout}
16573      */
16574     panel : false,
16575     
16576     /**
16577      * @property layout
16578      * for BC  - use el in new code
16579      * @type {Roo.Layout}
16580      */
16581     layout : false,
16582     
16583      /**
16584      * @cfg {Function|boolean} disabled
16585      * If this module is disabled by some rule, return true from the funtion
16586      */
16587     disabled : false,
16588     
16589     /**
16590      * @cfg {String} parent 
16591      * Name of parent element which it get xtype added to..
16592      */
16593     parent: false,
16594     
16595     /**
16596      * @cfg {String} order
16597      * Used to set the order in which elements are created (usefull for multiple tabs)
16598      */
16599     
16600     order : false,
16601     /**
16602      * @cfg {String} name
16603      * String to display while loading.
16604      */
16605     name : false,
16606     /**
16607      * @cfg {String} region
16608      * Region to render component to (defaults to center)
16609      */
16610     region : 'center',
16611     
16612     /**
16613      * @cfg {Array} items
16614      * A single item array - the first element is the root of the tree..
16615      * It's done this way to stay compatible with the Xtype system...
16616      */
16617     items : false,
16618     
16619     /**
16620      * @property _tree
16621      * The method that retuns the tree of parts that make up this compoennt 
16622      * @type {function}
16623      */
16624     _tree  : false,
16625     
16626      /**
16627      * render
16628      * render element to dom or tree
16629      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16630      */
16631     
16632     render : function(el)
16633     {
16634         
16635         el = el || false;
16636         var hp = this.parent ? 1 : 0;
16637         Roo.debug &&  Roo.log(this);
16638         
16639         var tree = this._tree ? this._tree() : this.tree();
16640
16641         
16642         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16643             // if parent is a '#.....' string, then let's use that..
16644             var ename = this.parent.substr(1);
16645             this.parent = false;
16646             Roo.debug && Roo.log(ename);
16647             switch (ename) {
16648                 case 'bootstrap-body':
16649                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16650                         // this is the BorderLayout standard?
16651                        this.parent = { el : true };
16652                        break;
16653                     }
16654                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16655                         // need to insert stuff...
16656                         this.parent =  {
16657                              el : new Roo.bootstrap.layout.Border({
16658                                  el : document.body, 
16659                      
16660                                  center: {
16661                                     titlebar: false,
16662                                     autoScroll:false,
16663                                     closeOnTab: true,
16664                                     tabPosition: 'top',
16665                                       //resizeTabs: true,
16666                                     alwaysShowTabs: true,
16667                                     hideTabs: false
16668                                      //minTabWidth: 140
16669                                  }
16670                              })
16671                         
16672                          };
16673                          break;
16674                     }
16675                          
16676                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16677                         this.parent = { el :  new  Roo.bootstrap.Body() };
16678                         Roo.debug && Roo.log("setting el to doc body");
16679                          
16680                     } else {
16681                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16682                     }
16683                     break;
16684                 case 'bootstrap':
16685                     this.parent = { el : true};
16686                     // fall through
16687                 default:
16688                     el = Roo.get(ename);
16689                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16690                         this.parent = { el : true};
16691                     }
16692                     
16693                     break;
16694             }
16695                 
16696             
16697             if (!el && !this.parent) {
16698                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16699                 return;
16700             }
16701         }
16702         
16703         Roo.debug && Roo.log("EL:");
16704         Roo.debug && Roo.log(el);
16705         Roo.debug && Roo.log("this.parent.el:");
16706         Roo.debug && Roo.log(this.parent.el);
16707         
16708
16709         // altertive root elements ??? - we need a better way to indicate these.
16710         var is_alt = Roo.XComponent.is_alt ||
16711                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16712                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16713                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16714         
16715         
16716         
16717         if (!this.parent && is_alt) {
16718             //el = Roo.get(document.body);
16719             this.parent = { el : true };
16720         }
16721             
16722             
16723         
16724         if (!this.parent) {
16725             
16726             Roo.debug && Roo.log("no parent - creating one");
16727             
16728             el = el ? Roo.get(el) : false;      
16729             
16730             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16731                 
16732                 this.parent =  {
16733                     el : new Roo.bootstrap.layout.Border({
16734                         el: el || document.body,
16735                     
16736                         center: {
16737                             titlebar: false,
16738                             autoScroll:false,
16739                             closeOnTab: true,
16740                             tabPosition: 'top',
16741                              //resizeTabs: true,
16742                             alwaysShowTabs: false,
16743                             hideTabs: true,
16744                             minTabWidth: 140,
16745                             overflow: 'visible'
16746                          }
16747                      })
16748                 };
16749             } else {
16750             
16751                 // it's a top level one..
16752                 this.parent =  {
16753                     el : new Roo.BorderLayout(el || document.body, {
16754                         center: {
16755                             titlebar: false,
16756                             autoScroll:false,
16757                             closeOnTab: true,
16758                             tabPosition: 'top',
16759                              //resizeTabs: true,
16760                             alwaysShowTabs: el && hp? false :  true,
16761                             hideTabs: el || !hp ? true :  false,
16762                             minTabWidth: 140
16763                          }
16764                     })
16765                 };
16766             }
16767         }
16768         
16769         if (!this.parent.el) {
16770                 // probably an old style ctor, which has been disabled.
16771                 return;
16772
16773         }
16774                 // The 'tree' method is  '_tree now' 
16775             
16776         tree.region = tree.region || this.region;
16777         var is_body = false;
16778         if (this.parent.el === true) {
16779             // bootstrap... - body..
16780             if (el) {
16781                 tree.el = el;
16782             }
16783             this.parent.el = Roo.factory(tree);
16784             is_body = true;
16785         }
16786         
16787         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16788         this.fireEvent('built', this);
16789         
16790         this.panel = this.el;
16791         this.layout = this.panel.layout;
16792         this.parentLayout = this.parent.layout  || false;  
16793          
16794     }
16795     
16796 });
16797
16798 Roo.apply(Roo.XComponent, {
16799     /**
16800      * @property  hideProgress
16801      * true to disable the building progress bar.. usefull on single page renders.
16802      * @type Boolean
16803      */
16804     hideProgress : false,
16805     /**
16806      * @property  buildCompleted
16807      * True when the builder has completed building the interface.
16808      * @type Boolean
16809      */
16810     buildCompleted : false,
16811      
16812     /**
16813      * @property  topModule
16814      * the upper most module - uses document.element as it's constructor.
16815      * @type Object
16816      */
16817      
16818     topModule  : false,
16819       
16820     /**
16821      * @property  modules
16822      * array of modules to be created by registration system.
16823      * @type {Array} of Roo.XComponent
16824      */
16825     
16826     modules : [],
16827     /**
16828      * @property  elmodules
16829      * array of modules to be created by which use #ID 
16830      * @type {Array} of Roo.XComponent
16831      */
16832      
16833     elmodules : [],
16834
16835      /**
16836      * @property  is_alt
16837      * Is an alternative Root - normally used by bootstrap or other systems,
16838      *    where the top element in the tree can wrap 'body' 
16839      * @type {boolean}  (default false)
16840      */
16841      
16842     is_alt : false,
16843     /**
16844      * @property  build_from_html
16845      * Build elements from html - used by bootstrap HTML stuff 
16846      *    - this is cleared after build is completed
16847      * @type {boolean}    (default false)
16848      */
16849      
16850     build_from_html : false,
16851     /**
16852      * Register components to be built later.
16853      *
16854      * This solves the following issues
16855      * - Building is not done on page load, but after an authentication process has occured.
16856      * - Interface elements are registered on page load
16857      * - Parent Interface elements may not be loaded before child, so this handles that..
16858      * 
16859      *
16860      * example:
16861      * 
16862      * MyApp.register({
16863           order : '000001',
16864           module : 'Pman.Tab.projectMgr',
16865           region : 'center',
16866           parent : 'Pman.layout',
16867           disabled : false,  // or use a function..
16868         })
16869      
16870      * * @param {Object} details about module
16871      */
16872     register : function(obj) {
16873                 
16874         Roo.XComponent.event.fireEvent('register', obj);
16875         switch(typeof(obj.disabled) ) {
16876                 
16877             case 'undefined':
16878                 break;
16879             
16880             case 'function':
16881                 if ( obj.disabled() ) {
16882                         return;
16883                 }
16884                 break;
16885             
16886             default:
16887                 if (obj.disabled || obj.region == '#disabled') {
16888                         return;
16889                 }
16890                 break;
16891         }
16892                 
16893         this.modules.push(obj);
16894          
16895     },
16896     /**
16897      * convert a string to an object..
16898      * eg. 'AAA.BBB' -> finds AAA.BBB
16899
16900      */
16901     
16902     toObject : function(str)
16903     {
16904         if (!str || typeof(str) == 'object') {
16905             return str;
16906         }
16907         if (str.substring(0,1) == '#') {
16908             return str;
16909         }
16910
16911         var ar = str.split('.');
16912         var rt, o;
16913         rt = ar.shift();
16914             /** eval:var:o */
16915         try {
16916             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16917         } catch (e) {
16918             throw "Module not found : " + str;
16919         }
16920         
16921         if (o === false) {
16922             throw "Module not found : " + str;
16923         }
16924         Roo.each(ar, function(e) {
16925             if (typeof(o[e]) == 'undefined') {
16926                 throw "Module not found : " + str;
16927             }
16928             o = o[e];
16929         });
16930         
16931         return o;
16932         
16933     },
16934     
16935     
16936     /**
16937      * move modules into their correct place in the tree..
16938      * 
16939      */
16940     preBuild : function ()
16941     {
16942         var _t = this;
16943         Roo.each(this.modules , function (obj)
16944         {
16945             Roo.XComponent.event.fireEvent('beforebuild', obj);
16946             
16947             var opar = obj.parent;
16948             try { 
16949                 obj.parent = this.toObject(opar);
16950             } catch(e) {
16951                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16952                 return;
16953             }
16954             
16955             if (!obj.parent) {
16956                 Roo.debug && Roo.log("GOT top level module");
16957                 Roo.debug && Roo.log(obj);
16958                 obj.modules = new Roo.util.MixedCollection(false, 
16959                     function(o) { return o.order + '' }
16960                 );
16961                 this.topModule = obj;
16962                 return;
16963             }
16964                         // parent is a string (usually a dom element name..)
16965             if (typeof(obj.parent) == 'string') {
16966                 this.elmodules.push(obj);
16967                 return;
16968             }
16969             if (obj.parent.constructor != Roo.XComponent) {
16970                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16971             }
16972             if (!obj.parent.modules) {
16973                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16974                     function(o) { return o.order + '' }
16975                 );
16976             }
16977             if (obj.parent.disabled) {
16978                 obj.disabled = true;
16979             }
16980             obj.parent.modules.add(obj);
16981         }, this);
16982     },
16983     
16984      /**
16985      * make a list of modules to build.
16986      * @return {Array} list of modules. 
16987      */ 
16988     
16989     buildOrder : function()
16990     {
16991         var _this = this;
16992         var cmp = function(a,b) {   
16993             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16994         };
16995         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16996             throw "No top level modules to build";
16997         }
16998         
16999         // make a flat list in order of modules to build.
17000         var mods = this.topModule ? [ this.topModule ] : [];
17001                 
17002         
17003         // elmodules (is a list of DOM based modules )
17004         Roo.each(this.elmodules, function(e) {
17005             mods.push(e);
17006             if (!this.topModule &&
17007                 typeof(e.parent) == 'string' &&
17008                 e.parent.substring(0,1) == '#' &&
17009                 Roo.get(e.parent.substr(1))
17010                ) {
17011                 
17012                 _this.topModule = e;
17013             }
17014             
17015         });
17016
17017         
17018         // add modules to their parents..
17019         var addMod = function(m) {
17020             Roo.debug && Roo.log("build Order: add: " + m.name);
17021                 
17022             mods.push(m);
17023             if (m.modules && !m.disabled) {
17024                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
17025                 m.modules.keySort('ASC',  cmp );
17026                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
17027     
17028                 m.modules.each(addMod);
17029             } else {
17030                 Roo.debug && Roo.log("build Order: no child modules");
17031             }
17032             // not sure if this is used any more..
17033             if (m.finalize) {
17034                 m.finalize.name = m.name + " (clean up) ";
17035                 mods.push(m.finalize);
17036             }
17037             
17038         }
17039         if (this.topModule && this.topModule.modules) { 
17040             this.topModule.modules.keySort('ASC',  cmp );
17041             this.topModule.modules.each(addMod);
17042         } 
17043         return mods;
17044     },
17045     
17046      /**
17047      * Build the registered modules.
17048      * @param {Object} parent element.
17049      * @param {Function} optional method to call after module has been added.
17050      * 
17051      */ 
17052    
17053     build : function(opts) 
17054     {
17055         
17056         if (typeof(opts) != 'undefined') {
17057             Roo.apply(this,opts);
17058         }
17059         
17060         this.preBuild();
17061         var mods = this.buildOrder();
17062       
17063         //this.allmods = mods;
17064         //Roo.debug && Roo.log(mods);
17065         //return;
17066         if (!mods.length) { // should not happen
17067             throw "NO modules!!!";
17068         }
17069         
17070         
17071         var msg = "Building Interface...";
17072         // flash it up as modal - so we store the mask!?
17073         if (!this.hideProgress && Roo.MessageBox) {
17074             Roo.MessageBox.show({ title: 'loading' });
17075             Roo.MessageBox.show({
17076                title: "Please wait...",
17077                msg: msg,
17078                width:450,
17079                progress:true,
17080                buttons : false,
17081                closable:false,
17082                modal: false
17083               
17084             });
17085         }
17086         var total = mods.length;
17087         
17088         var _this = this;
17089         var progressRun = function() {
17090             if (!mods.length) {
17091                 Roo.debug && Roo.log('hide?');
17092                 if (!this.hideProgress && Roo.MessageBox) {
17093                     Roo.MessageBox.hide();
17094                 }
17095                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
17096                 
17097                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
17098                 
17099                 // THE END...
17100                 return false;   
17101             }
17102             
17103             var m = mods.shift();
17104             
17105             
17106             Roo.debug && Roo.log(m);
17107             // not sure if this is supported any more.. - modules that are are just function
17108             if (typeof(m) == 'function') { 
17109                 m.call(this);
17110                 return progressRun.defer(10, _this);
17111             } 
17112             
17113             
17114             msg = "Building Interface " + (total  - mods.length) + 
17115                     " of " + total + 
17116                     (m.name ? (' - ' + m.name) : '');
17117                         Roo.debug && Roo.log(msg);
17118             if (!_this.hideProgress &&  Roo.MessageBox) { 
17119                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
17120             }
17121             
17122          
17123             // is the module disabled?
17124             var disabled = (typeof(m.disabled) == 'function') ?
17125                 m.disabled.call(m.module.disabled) : m.disabled;    
17126             
17127             
17128             if (disabled) {
17129                 return progressRun(); // we do not update the display!
17130             }
17131             
17132             // now build 
17133             
17134                         
17135                         
17136             m.render();
17137             // it's 10 on top level, and 1 on others??? why...
17138             return progressRun.defer(10, _this);
17139              
17140         }
17141         progressRun.defer(1, _this);
17142      
17143         
17144         
17145     },
17146     /**
17147      * Overlay a set of modified strings onto a component
17148      * This is dependant on our builder exporting the strings and 'named strings' elements.
17149      * 
17150      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
17151      * @param {Object} associative array of 'named' string and it's new value.
17152      * 
17153      */
17154         overlayStrings : function( component, strings )
17155     {
17156         if (typeof(component['_named_strings']) == 'undefined') {
17157             throw "ERROR: component does not have _named_strings";
17158         }
17159         for ( var k in strings ) {
17160             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
17161             if (md !== false) {
17162                 component['_strings'][md] = strings[k];
17163             } else {
17164                 Roo.log('could not find named string: ' + k + ' in');
17165                 Roo.log(component);
17166             }
17167             
17168         }
17169         
17170     },
17171     
17172         
17173         /**
17174          * Event Object.
17175          *
17176          *
17177          */
17178         event: false, 
17179     /**
17180          * wrapper for event.on - aliased later..  
17181          * Typically use to register a event handler for register:
17182          *
17183          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
17184          *
17185          */
17186     on : false
17187    
17188     
17189     
17190 });
17191
17192 Roo.XComponent.event = new Roo.util.Observable({
17193                 events : { 
17194                         /**
17195                          * @event register
17196                          * Fires when an Component is registered,
17197                          * set the disable property on the Component to stop registration.
17198                          * @param {Roo.XComponent} c the component being registerd.
17199                          * 
17200                          */
17201                         'register' : true,
17202             /**
17203                          * @event beforebuild
17204                          * Fires before each Component is built
17205                          * can be used to apply permissions.
17206                          * @param {Roo.XComponent} c the component being registerd.
17207                          * 
17208                          */
17209                         'beforebuild' : true,
17210                         /**
17211                          * @event buildcomplete
17212                          * Fires on the top level element when all elements have been built
17213                          * @param {Roo.XComponent} the top level component.
17214                          */
17215                         'buildcomplete' : true
17216                         
17217                 }
17218 });
17219
17220 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
17221  //
17222  /**
17223  * marked - a markdown parser
17224  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
17225  * https://github.com/chjj/marked
17226  */
17227
17228
17229 /**
17230  *
17231  * Roo.Markdown - is a very crude wrapper around marked..
17232  *
17233  * usage:
17234  * 
17235  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
17236  * 
17237  * Note: move the sample code to the bottom of this
17238  * file before uncommenting it.
17239  *
17240  */
17241
17242 Roo.Markdown = {};
17243 Roo.Markdown.toHtml = function(text) {
17244     
17245     var c = new Roo.Markdown.marked.setOptions({
17246             renderer: new Roo.Markdown.marked.Renderer(),
17247             gfm: true,
17248             tables: true,
17249             breaks: false,
17250             pedantic: false,
17251             sanitize: false,
17252             smartLists: true,
17253             smartypants: false
17254           });
17255     // A FEW HACKS!!?
17256     
17257     text = text.replace(/\\\n/g,' ');
17258     return Roo.Markdown.marked(text);
17259 };
17260 //
17261 // converter
17262 //
17263 // Wraps all "globals" so that the only thing
17264 // exposed is makeHtml().
17265 //
17266 (function() {
17267     
17268      /**
17269          * eval:var:escape
17270          * eval:var:unescape
17271          * eval:var:replace
17272          */
17273       
17274     /**
17275      * Helpers
17276      */
17277     
17278     var escape = function (html, encode) {
17279       return html
17280         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
17281         .replace(/</g, '&lt;')
17282         .replace(/>/g, '&gt;')
17283         .replace(/"/g, '&quot;')
17284         .replace(/'/g, '&#39;');
17285     }
17286     
17287     var unescape = function (html) {
17288         // explicitly match decimal, hex, and named HTML entities 
17289       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
17290         n = n.toLowerCase();
17291         if (n === 'colon') { return ':'; }
17292         if (n.charAt(0) === '#') {
17293           return n.charAt(1) === 'x'
17294             ? String.fromCharCode(parseInt(n.substring(2), 16))
17295             : String.fromCharCode(+n.substring(1));
17296         }
17297         return '';
17298       });
17299     }
17300     
17301     var replace = function (regex, opt) {
17302       regex = regex.source;
17303       opt = opt || '';
17304       return function self(name, val) {
17305         if (!name) { return new RegExp(regex, opt); }
17306         val = val.source || val;
17307         val = val.replace(/(^|[^\[])\^/g, '$1');
17308         regex = regex.replace(name, val);
17309         return self;
17310       };
17311     }
17312
17313
17314          /**
17315          * eval:var:noop
17316     */
17317     var noop = function () {}
17318     noop.exec = noop;
17319     
17320          /**
17321          * eval:var:merge
17322     */
17323     var merge = function (obj) {
17324       var i = 1
17325         , target
17326         , key;
17327     
17328       for (; i < arguments.length; i++) {
17329         target = arguments[i];
17330         for (key in target) {
17331           if (Object.prototype.hasOwnProperty.call(target, key)) {
17332             obj[key] = target[key];
17333           }
17334         }
17335       }
17336     
17337       return obj;
17338     }
17339     
17340     
17341     /**
17342      * Block-Level Grammar
17343      */
17344     
17345     
17346     
17347     
17348     var block = {
17349       newline: /^\n+/,
17350       code: /^( {4}[^\n]+\n*)+/,
17351       fences: noop,
17352       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
17353       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
17354       nptable: noop,
17355       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
17356       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
17357       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
17358       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
17359       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
17360       table: noop,
17361       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
17362       text: /^[^\n]+/
17363     };
17364     
17365     block.bullet = /(?:[*+-]|\d+\.)/;
17366     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
17367     block.item = replace(block.item, 'gm')
17368       (/bull/g, block.bullet)
17369       ();
17370     
17371     block.list = replace(block.list)
17372       (/bull/g, block.bullet)
17373       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
17374       ('def', '\\n+(?=' + block.def.source + ')')
17375       ();
17376     
17377     block.blockquote = replace(block.blockquote)
17378       ('def', block.def)
17379       ();
17380     
17381     block._tag = '(?!(?:'
17382       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
17383       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
17384       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
17385     
17386     block.html = replace(block.html)
17387       ('comment', /<!--[\s\S]*?-->/)
17388       ('closed', /<(tag)[\s\S]+?<\/\1>/)
17389       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
17390       (/tag/g, block._tag)
17391       ();
17392     
17393     block.paragraph = replace(block.paragraph)
17394       ('hr', block.hr)
17395       ('heading', block.heading)
17396       ('lheading', block.lheading)
17397       ('blockquote', block.blockquote)
17398       ('tag', '<' + block._tag)
17399       ('def', block.def)
17400       ();
17401     
17402     /**
17403      * Normal Block Grammar
17404      */
17405     
17406     block.normal = merge({}, block);
17407     
17408     /**
17409      * GFM Block Grammar
17410      */
17411     
17412     block.gfm = merge({}, block.normal, {
17413       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
17414       paragraph: /^/,
17415       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
17416     });
17417     
17418     block.gfm.paragraph = replace(block.paragraph)
17419       ('(?!', '(?!'
17420         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
17421         + block.list.source.replace('\\1', '\\3') + '|')
17422       ();
17423     
17424     /**
17425      * GFM + Tables Block Grammar
17426      */
17427     
17428     block.tables = merge({}, block.gfm, {
17429       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
17430       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
17431     });
17432     
17433     /**
17434      * Block Lexer
17435      */
17436     
17437     var Lexer = function (options) {
17438       this.tokens = [];
17439       this.tokens.links = {};
17440       this.options = options || marked.defaults;
17441       this.rules = block.normal;
17442     
17443       if (this.options.gfm) {
17444         if (this.options.tables) {
17445           this.rules = block.tables;
17446         } else {
17447           this.rules = block.gfm;
17448         }
17449       }
17450     }
17451     
17452     /**
17453      * Expose Block Rules
17454      */
17455     
17456     Lexer.rules = block;
17457     
17458     /**
17459      * Static Lex Method
17460      */
17461     
17462     Lexer.lex = function(src, options) {
17463       var lexer = new Lexer(options);
17464       return lexer.lex(src);
17465     };
17466     
17467     /**
17468      * Preprocessing
17469      */
17470     
17471     Lexer.prototype.lex = function(src) {
17472       src = src
17473         .replace(/\r\n|\r/g, '\n')
17474         .replace(/\t/g, '    ')
17475         .replace(/\u00a0/g, ' ')
17476         .replace(/\u2424/g, '\n');
17477     
17478       return this.token(src, true);
17479     };
17480     
17481     /**
17482      * Lexing
17483      */
17484     
17485     Lexer.prototype.token = function(src, top, bq) {
17486       var src = src.replace(/^ +$/gm, '')
17487         , next
17488         , loose
17489         , cap
17490         , bull
17491         , b
17492         , item
17493         , space
17494         , i
17495         , l;
17496     
17497       while (src) {
17498         // newline
17499         if (cap = this.rules.newline.exec(src)) {
17500           src = src.substring(cap[0].length);
17501           if (cap[0].length > 1) {
17502             this.tokens.push({
17503               type: 'space'
17504             });
17505           }
17506         }
17507     
17508         // code
17509         if (cap = this.rules.code.exec(src)) {
17510           src = src.substring(cap[0].length);
17511           cap = cap[0].replace(/^ {4}/gm, '');
17512           this.tokens.push({
17513             type: 'code',
17514             text: !this.options.pedantic
17515               ? cap.replace(/\n+$/, '')
17516               : cap
17517           });
17518           continue;
17519         }
17520     
17521         // fences (gfm)
17522         if (cap = this.rules.fences.exec(src)) {
17523           src = src.substring(cap[0].length);
17524           this.tokens.push({
17525             type: 'code',
17526             lang: cap[2],
17527             text: cap[3] || ''
17528           });
17529           continue;
17530         }
17531     
17532         // heading
17533         if (cap = this.rules.heading.exec(src)) {
17534           src = src.substring(cap[0].length);
17535           this.tokens.push({
17536             type: 'heading',
17537             depth: cap[1].length,
17538             text: cap[2]
17539           });
17540           continue;
17541         }
17542     
17543         // table no leading pipe (gfm)
17544         if (top && (cap = this.rules.nptable.exec(src))) {
17545           src = src.substring(cap[0].length);
17546     
17547           item = {
17548             type: 'table',
17549             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17550             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17551             cells: cap[3].replace(/\n$/, '').split('\n')
17552           };
17553     
17554           for (i = 0; i < item.align.length; i++) {
17555             if (/^ *-+: *$/.test(item.align[i])) {
17556               item.align[i] = 'right';
17557             } else if (/^ *:-+: *$/.test(item.align[i])) {
17558               item.align[i] = 'center';
17559             } else if (/^ *:-+ *$/.test(item.align[i])) {
17560               item.align[i] = 'left';
17561             } else {
17562               item.align[i] = null;
17563             }
17564           }
17565     
17566           for (i = 0; i < item.cells.length; i++) {
17567             item.cells[i] = item.cells[i].split(/ *\| */);
17568           }
17569     
17570           this.tokens.push(item);
17571     
17572           continue;
17573         }
17574     
17575         // lheading
17576         if (cap = this.rules.lheading.exec(src)) {
17577           src = src.substring(cap[0].length);
17578           this.tokens.push({
17579             type: 'heading',
17580             depth: cap[2] === '=' ? 1 : 2,
17581             text: cap[1]
17582           });
17583           continue;
17584         }
17585     
17586         // hr
17587         if (cap = this.rules.hr.exec(src)) {
17588           src = src.substring(cap[0].length);
17589           this.tokens.push({
17590             type: 'hr'
17591           });
17592           continue;
17593         }
17594     
17595         // blockquote
17596         if (cap = this.rules.blockquote.exec(src)) {
17597           src = src.substring(cap[0].length);
17598     
17599           this.tokens.push({
17600             type: 'blockquote_start'
17601           });
17602     
17603           cap = cap[0].replace(/^ *> ?/gm, '');
17604     
17605           // Pass `top` to keep the current
17606           // "toplevel" state. This is exactly
17607           // how markdown.pl works.
17608           this.token(cap, top, true);
17609     
17610           this.tokens.push({
17611             type: 'blockquote_end'
17612           });
17613     
17614           continue;
17615         }
17616     
17617         // list
17618         if (cap = this.rules.list.exec(src)) {
17619           src = src.substring(cap[0].length);
17620           bull = cap[2];
17621     
17622           this.tokens.push({
17623             type: 'list_start',
17624             ordered: bull.length > 1
17625           });
17626     
17627           // Get each top-level item.
17628           cap = cap[0].match(this.rules.item);
17629     
17630           next = false;
17631           l = cap.length;
17632           i = 0;
17633     
17634           for (; i < l; i++) {
17635             item = cap[i];
17636     
17637             // Remove the list item's bullet
17638             // so it is seen as the next token.
17639             space = item.length;
17640             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17641     
17642             // Outdent whatever the
17643             // list item contains. Hacky.
17644             if (~item.indexOf('\n ')) {
17645               space -= item.length;
17646               item = !this.options.pedantic
17647                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17648                 : item.replace(/^ {1,4}/gm, '');
17649             }
17650     
17651             // Determine whether the next list item belongs here.
17652             // Backpedal if it does not belong in this list.
17653             if (this.options.smartLists && i !== l - 1) {
17654               b = block.bullet.exec(cap[i + 1])[0];
17655               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17656                 src = cap.slice(i + 1).join('\n') + src;
17657                 i = l - 1;
17658               }
17659             }
17660     
17661             // Determine whether item is loose or not.
17662             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17663             // for discount behavior.
17664             loose = next || /\n\n(?!\s*$)/.test(item);
17665             if (i !== l - 1) {
17666               next = item.charAt(item.length - 1) === '\n';
17667               if (!loose) { loose = next; }
17668             }
17669     
17670             this.tokens.push({
17671               type: loose
17672                 ? 'loose_item_start'
17673                 : 'list_item_start'
17674             });
17675     
17676             // Recurse.
17677             this.token(item, false, bq);
17678     
17679             this.tokens.push({
17680               type: 'list_item_end'
17681             });
17682           }
17683     
17684           this.tokens.push({
17685             type: 'list_end'
17686           });
17687     
17688           continue;
17689         }
17690     
17691         // html
17692         if (cap = this.rules.html.exec(src)) {
17693           src = src.substring(cap[0].length);
17694           this.tokens.push({
17695             type: this.options.sanitize
17696               ? 'paragraph'
17697               : 'html',
17698             pre: !this.options.sanitizer
17699               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17700             text: cap[0]
17701           });
17702           continue;
17703         }
17704     
17705         // def
17706         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17707           src = src.substring(cap[0].length);
17708           this.tokens.links[cap[1].toLowerCase()] = {
17709             href: cap[2],
17710             title: cap[3]
17711           };
17712           continue;
17713         }
17714     
17715         // table (gfm)
17716         if (top && (cap = this.rules.table.exec(src))) {
17717           src = src.substring(cap[0].length);
17718     
17719           item = {
17720             type: 'table',
17721             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17722             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17723             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17724           };
17725     
17726           for (i = 0; i < item.align.length; i++) {
17727             if (/^ *-+: *$/.test(item.align[i])) {
17728               item.align[i] = 'right';
17729             } else if (/^ *:-+: *$/.test(item.align[i])) {
17730               item.align[i] = 'center';
17731             } else if (/^ *:-+ *$/.test(item.align[i])) {
17732               item.align[i] = 'left';
17733             } else {
17734               item.align[i] = null;
17735             }
17736           }
17737     
17738           for (i = 0; i < item.cells.length; i++) {
17739             item.cells[i] = item.cells[i]
17740               .replace(/^ *\| *| *\| *$/g, '')
17741               .split(/ *\| */);
17742           }
17743     
17744           this.tokens.push(item);
17745     
17746           continue;
17747         }
17748     
17749         // top-level paragraph
17750         if (top && (cap = this.rules.paragraph.exec(src))) {
17751           src = src.substring(cap[0].length);
17752           this.tokens.push({
17753             type: 'paragraph',
17754             text: cap[1].charAt(cap[1].length - 1) === '\n'
17755               ? cap[1].slice(0, -1)
17756               : cap[1]
17757           });
17758           continue;
17759         }
17760     
17761         // text
17762         if (cap = this.rules.text.exec(src)) {
17763           // Top-level should never reach here.
17764           src = src.substring(cap[0].length);
17765           this.tokens.push({
17766             type: 'text',
17767             text: cap[0]
17768           });
17769           continue;
17770         }
17771     
17772         if (src) {
17773           throw new
17774             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17775         }
17776       }
17777     
17778       return this.tokens;
17779     };
17780     
17781     /**
17782      * Inline-Level Grammar
17783      */
17784     
17785     var inline = {
17786       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17787       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17788       url: noop,
17789       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17790       link: /^!?\[(inside)\]\(href\)/,
17791       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17792       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17793       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17794       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17795       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17796       br: /^ {2,}\n(?!\s*$)/,
17797       del: noop,
17798       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17799     };
17800     
17801     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17802     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17803     
17804     inline.link = replace(inline.link)
17805       ('inside', inline._inside)
17806       ('href', inline._href)
17807       ();
17808     
17809     inline.reflink = replace(inline.reflink)
17810       ('inside', inline._inside)
17811       ();
17812     
17813     /**
17814      * Normal Inline Grammar
17815      */
17816     
17817     inline.normal = merge({}, inline);
17818     
17819     /**
17820      * Pedantic Inline Grammar
17821      */
17822     
17823     inline.pedantic = merge({}, inline.normal, {
17824       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17825       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17826     });
17827     
17828     /**
17829      * GFM Inline Grammar
17830      */
17831     
17832     inline.gfm = merge({}, inline.normal, {
17833       escape: replace(inline.escape)('])', '~|])')(),
17834       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17835       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17836       text: replace(inline.text)
17837         (']|', '~]|')
17838         ('|', '|https?://|')
17839         ()
17840     });
17841     
17842     /**
17843      * GFM + Line Breaks Inline Grammar
17844      */
17845     
17846     inline.breaks = merge({}, inline.gfm, {
17847       br: replace(inline.br)('{2,}', '*')(),
17848       text: replace(inline.gfm.text)('{2,}', '*')()
17849     });
17850     
17851     /**
17852      * Inline Lexer & Compiler
17853      */
17854     
17855     var InlineLexer  = function (links, options) {
17856       this.options = options || marked.defaults;
17857       this.links = links;
17858       this.rules = inline.normal;
17859       this.renderer = this.options.renderer || new Renderer;
17860       this.renderer.options = this.options;
17861     
17862       if (!this.links) {
17863         throw new
17864           Error('Tokens array requires a `links` property.');
17865       }
17866     
17867       if (this.options.gfm) {
17868         if (this.options.breaks) {
17869           this.rules = inline.breaks;
17870         } else {
17871           this.rules = inline.gfm;
17872         }
17873       } else if (this.options.pedantic) {
17874         this.rules = inline.pedantic;
17875       }
17876     }
17877     
17878     /**
17879      * Expose Inline Rules
17880      */
17881     
17882     InlineLexer.rules = inline;
17883     
17884     /**
17885      * Static Lexing/Compiling Method
17886      */
17887     
17888     InlineLexer.output = function(src, links, options) {
17889       var inline = new InlineLexer(links, options);
17890       return inline.output(src);
17891     };
17892     
17893     /**
17894      * Lexing/Compiling
17895      */
17896     
17897     InlineLexer.prototype.output = function(src) {
17898       var out = ''
17899         , link
17900         , text
17901         , href
17902         , cap;
17903     
17904       while (src) {
17905         // escape
17906         if (cap = this.rules.escape.exec(src)) {
17907           src = src.substring(cap[0].length);
17908           out += cap[1];
17909           continue;
17910         }
17911     
17912         // autolink
17913         if (cap = this.rules.autolink.exec(src)) {
17914           src = src.substring(cap[0].length);
17915           if (cap[2] === '@') {
17916             text = cap[1].charAt(6) === ':'
17917               ? this.mangle(cap[1].substring(7))
17918               : this.mangle(cap[1]);
17919             href = this.mangle('mailto:') + text;
17920           } else {
17921             text = escape(cap[1]);
17922             href = text;
17923           }
17924           out += this.renderer.link(href, null, text);
17925           continue;
17926         }
17927     
17928         // url (gfm)
17929         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17930           src = src.substring(cap[0].length);
17931           text = escape(cap[1]);
17932           href = text;
17933           out += this.renderer.link(href, null, text);
17934           continue;
17935         }
17936     
17937         // tag
17938         if (cap = this.rules.tag.exec(src)) {
17939           if (!this.inLink && /^<a /i.test(cap[0])) {
17940             this.inLink = true;
17941           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17942             this.inLink = false;
17943           }
17944           src = src.substring(cap[0].length);
17945           out += this.options.sanitize
17946             ? this.options.sanitizer
17947               ? this.options.sanitizer(cap[0])
17948               : escape(cap[0])
17949             : cap[0];
17950           continue;
17951         }
17952     
17953         // link
17954         if (cap = this.rules.link.exec(src)) {
17955           src = src.substring(cap[0].length);
17956           this.inLink = true;
17957           out += this.outputLink(cap, {
17958             href: cap[2],
17959             title: cap[3]
17960           });
17961           this.inLink = false;
17962           continue;
17963         }
17964     
17965         // reflink, nolink
17966         if ((cap = this.rules.reflink.exec(src))
17967             || (cap = this.rules.nolink.exec(src))) {
17968           src = src.substring(cap[0].length);
17969           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17970           link = this.links[link.toLowerCase()];
17971           if (!link || !link.href) {
17972             out += cap[0].charAt(0);
17973             src = cap[0].substring(1) + src;
17974             continue;
17975           }
17976           this.inLink = true;
17977           out += this.outputLink(cap, link);
17978           this.inLink = false;
17979           continue;
17980         }
17981     
17982         // strong
17983         if (cap = this.rules.strong.exec(src)) {
17984           src = src.substring(cap[0].length);
17985           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17986           continue;
17987         }
17988     
17989         // em
17990         if (cap = this.rules.em.exec(src)) {
17991           src = src.substring(cap[0].length);
17992           out += this.renderer.em(this.output(cap[2] || cap[1]));
17993           continue;
17994         }
17995     
17996         // code
17997         if (cap = this.rules.code.exec(src)) {
17998           src = src.substring(cap[0].length);
17999           out += this.renderer.codespan(escape(cap[2], true));
18000           continue;
18001         }
18002     
18003         // br
18004         if (cap = this.rules.br.exec(src)) {
18005           src = src.substring(cap[0].length);
18006           out += this.renderer.br();
18007           continue;
18008         }
18009     
18010         // del (gfm)
18011         if (cap = this.rules.del.exec(src)) {
18012           src = src.substring(cap[0].length);
18013           out += this.renderer.del(this.output(cap[1]));
18014           continue;
18015         }
18016     
18017         // text
18018         if (cap = this.rules.text.exec(src)) {
18019           src = src.substring(cap[0].length);
18020           out += this.renderer.text(escape(this.smartypants(cap[0])));
18021           continue;
18022         }
18023     
18024         if (src) {
18025           throw new
18026             Error('Infinite loop on byte: ' + src.charCodeAt(0));
18027         }
18028       }
18029     
18030       return out;
18031     };
18032     
18033     /**
18034      * Compile Link
18035      */
18036     
18037     InlineLexer.prototype.outputLink = function(cap, link) {
18038       var href = escape(link.href)
18039         , title = link.title ? escape(link.title) : null;
18040     
18041       return cap[0].charAt(0) !== '!'
18042         ? this.renderer.link(href, title, this.output(cap[1]))
18043         : this.renderer.image(href, title, escape(cap[1]));
18044     };
18045     
18046     /**
18047      * Smartypants Transformations
18048      */
18049     
18050     InlineLexer.prototype.smartypants = function(text) {
18051       if (!this.options.smartypants)  { return text; }
18052       return text
18053         // em-dashes
18054         .replace(/---/g, '\u2014')
18055         // en-dashes
18056         .replace(/--/g, '\u2013')
18057         // opening singles
18058         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
18059         // closing singles & apostrophes
18060         .replace(/'/g, '\u2019')
18061         // opening doubles
18062         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
18063         // closing doubles
18064         .replace(/"/g, '\u201d')
18065         // ellipses
18066         .replace(/\.{3}/g, '\u2026');
18067     };
18068     
18069     /**
18070      * Mangle Links
18071      */
18072     
18073     InlineLexer.prototype.mangle = function(text) {
18074       if (!this.options.mangle) { return text; }
18075       var out = ''
18076         , l = text.length
18077         , i = 0
18078         , ch;
18079     
18080       for (; i < l; i++) {
18081         ch = text.charCodeAt(i);
18082         if (Math.random() > 0.5) {
18083           ch = 'x' + ch.toString(16);
18084         }
18085         out += '&#' + ch + ';';
18086       }
18087     
18088       return out;
18089     };
18090     
18091     /**
18092      * Renderer
18093      */
18094     
18095      /**
18096          * eval:var:Renderer
18097     */
18098     
18099     var Renderer   = function (options) {
18100       this.options = options || {};
18101     }
18102     
18103     Renderer.prototype.code = function(code, lang, escaped) {
18104       if (this.options.highlight) {
18105         var out = this.options.highlight(code, lang);
18106         if (out != null && out !== code) {
18107           escaped = true;
18108           code = out;
18109         }
18110       } else {
18111             // hack!!! - it's already escapeD?
18112             escaped = true;
18113       }
18114     
18115       if (!lang) {
18116         return '<pre><code>'
18117           + (escaped ? code : escape(code, true))
18118           + '\n</code></pre>';
18119       }
18120     
18121       return '<pre><code class="'
18122         + this.options.langPrefix
18123         + escape(lang, true)
18124         + '">'
18125         + (escaped ? code : escape(code, true))
18126         + '\n</code></pre>\n';
18127     };
18128     
18129     Renderer.prototype.blockquote = function(quote) {
18130       return '<blockquote>\n' + quote + '</blockquote>\n';
18131     };
18132     
18133     Renderer.prototype.html = function(html) {
18134       return html;
18135     };
18136     
18137     Renderer.prototype.heading = function(text, level, raw) {
18138       return '<h'
18139         + level
18140         + ' id="'
18141         + this.options.headerPrefix
18142         + raw.toLowerCase().replace(/[^\w]+/g, '-')
18143         + '">'
18144         + text
18145         + '</h'
18146         + level
18147         + '>\n';
18148     };
18149     
18150     Renderer.prototype.hr = function() {
18151       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
18152     };
18153     
18154     Renderer.prototype.list = function(body, ordered) {
18155       var type = ordered ? 'ol' : 'ul';
18156       return '<' + type + '>\n' + body + '</' + type + '>\n';
18157     };
18158     
18159     Renderer.prototype.listitem = function(text) {
18160       return '<li>' + text + '</li>\n';
18161     };
18162     
18163     Renderer.prototype.paragraph = function(text) {
18164       return '<p>' + text + '</p>\n';
18165     };
18166     
18167     Renderer.prototype.table = function(header, body) {
18168       return '<table class="table table-striped">\n'
18169         + '<thead>\n'
18170         + header
18171         + '</thead>\n'
18172         + '<tbody>\n'
18173         + body
18174         + '</tbody>\n'
18175         + '</table>\n';
18176     };
18177     
18178     Renderer.prototype.tablerow = function(content) {
18179       return '<tr>\n' + content + '</tr>\n';
18180     };
18181     
18182     Renderer.prototype.tablecell = function(content, flags) {
18183       var type = flags.header ? 'th' : 'td';
18184       var tag = flags.align
18185         ? '<' + type + ' style="text-align:' + flags.align + '">'
18186         : '<' + type + '>';
18187       return tag + content + '</' + type + '>\n';
18188     };
18189     
18190     // span level renderer
18191     Renderer.prototype.strong = function(text) {
18192       return '<strong>' + text + '</strong>';
18193     };
18194     
18195     Renderer.prototype.em = function(text) {
18196       return '<em>' + text + '</em>';
18197     };
18198     
18199     Renderer.prototype.codespan = function(text) {
18200       return '<code>' + text + '</code>';
18201     };
18202     
18203     Renderer.prototype.br = function() {
18204       return this.options.xhtml ? '<br/>' : '<br>';
18205     };
18206     
18207     Renderer.prototype.del = function(text) {
18208       return '<del>' + text + '</del>';
18209     };
18210     
18211     Renderer.prototype.link = function(href, title, text) {
18212       if (this.options.sanitize) {
18213         try {
18214           var prot = decodeURIComponent(unescape(href))
18215             .replace(/[^\w:]/g, '')
18216             .toLowerCase();
18217         } catch (e) {
18218           return '';
18219         }
18220         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
18221           return '';
18222         }
18223       }
18224       var out = '<a href="' + href + '"';
18225       if (title) {
18226         out += ' title="' + title + '"';
18227       }
18228       out += '>' + text + '</a>';
18229       return out;
18230     };
18231     
18232     Renderer.prototype.image = function(href, title, text) {
18233       var out = '<img src="' + href + '" alt="' + text + '"';
18234       if (title) {
18235         out += ' title="' + title + '"';
18236       }
18237       out += this.options.xhtml ? '/>' : '>';
18238       return out;
18239     };
18240     
18241     Renderer.prototype.text = function(text) {
18242       return text;
18243     };
18244     
18245     /**
18246      * Parsing & Compiling
18247      */
18248          /**
18249          * eval:var:Parser
18250     */
18251     
18252     var Parser= function (options) {
18253       this.tokens = [];
18254       this.token = null;
18255       this.options = options || marked.defaults;
18256       this.options.renderer = this.options.renderer || new Renderer;
18257       this.renderer = this.options.renderer;
18258       this.renderer.options = this.options;
18259     }
18260     
18261     /**
18262      * Static Parse Method
18263      */
18264     
18265     Parser.parse = function(src, options, renderer) {
18266       var parser = new Parser(options, renderer);
18267       return parser.parse(src);
18268     };
18269     
18270     /**
18271      * Parse Loop
18272      */
18273     
18274     Parser.prototype.parse = function(src) {
18275       this.inline = new InlineLexer(src.links, this.options, this.renderer);
18276       this.tokens = src.reverse();
18277     
18278       var out = '';
18279       while (this.next()) {
18280         out += this.tok();
18281       }
18282     
18283       return out;
18284     };
18285     
18286     /**
18287      * Next Token
18288      */
18289     
18290     Parser.prototype.next = function() {
18291       return this.token = this.tokens.pop();
18292     };
18293     
18294     /**
18295      * Preview Next Token
18296      */
18297     
18298     Parser.prototype.peek = function() {
18299       return this.tokens[this.tokens.length - 1] || 0;
18300     };
18301     
18302     /**
18303      * Parse Text Tokens
18304      */
18305     
18306     Parser.prototype.parseText = function() {
18307       var body = this.token.text;
18308     
18309       while (this.peek().type === 'text') {
18310         body += '\n' + this.next().text;
18311       }
18312     
18313       return this.inline.output(body);
18314     };
18315     
18316     /**
18317      * Parse Current Token
18318      */
18319     
18320     Parser.prototype.tok = function() {
18321       switch (this.token.type) {
18322         case 'space': {
18323           return '';
18324         }
18325         case 'hr': {
18326           return this.renderer.hr();
18327         }
18328         case 'heading': {
18329           return this.renderer.heading(
18330             this.inline.output(this.token.text),
18331             this.token.depth,
18332             this.token.text);
18333         }
18334         case 'code': {
18335           return this.renderer.code(this.token.text,
18336             this.token.lang,
18337             this.token.escaped);
18338         }
18339         case 'table': {
18340           var header = ''
18341             , body = ''
18342             , i
18343             , row
18344             , cell
18345             , flags
18346             , j;
18347     
18348           // header
18349           cell = '';
18350           for (i = 0; i < this.token.header.length; i++) {
18351             flags = { header: true, align: this.token.align[i] };
18352             cell += this.renderer.tablecell(
18353               this.inline.output(this.token.header[i]),
18354               { header: true, align: this.token.align[i] }
18355             );
18356           }
18357           header += this.renderer.tablerow(cell);
18358     
18359           for (i = 0; i < this.token.cells.length; i++) {
18360             row = this.token.cells[i];
18361     
18362             cell = '';
18363             for (j = 0; j < row.length; j++) {
18364               cell += this.renderer.tablecell(
18365                 this.inline.output(row[j]),
18366                 { header: false, align: this.token.align[j] }
18367               );
18368             }
18369     
18370             body += this.renderer.tablerow(cell);
18371           }
18372           return this.renderer.table(header, body);
18373         }
18374         case 'blockquote_start': {
18375           var body = '';
18376     
18377           while (this.next().type !== 'blockquote_end') {
18378             body += this.tok();
18379           }
18380     
18381           return this.renderer.blockquote(body);
18382         }
18383         case 'list_start': {
18384           var body = ''
18385             , ordered = this.token.ordered;
18386     
18387           while (this.next().type !== 'list_end') {
18388             body += this.tok();
18389           }
18390     
18391           return this.renderer.list(body, ordered);
18392         }
18393         case 'list_item_start': {
18394           var body = '';
18395     
18396           while (this.next().type !== 'list_item_end') {
18397             body += this.token.type === 'text'
18398               ? this.parseText()
18399               : this.tok();
18400           }
18401     
18402           return this.renderer.listitem(body);
18403         }
18404         case 'loose_item_start': {
18405           var body = '';
18406     
18407           while (this.next().type !== 'list_item_end') {
18408             body += this.tok();
18409           }
18410     
18411           return this.renderer.listitem(body);
18412         }
18413         case 'html': {
18414           var html = !this.token.pre && !this.options.pedantic
18415             ? this.inline.output(this.token.text)
18416             : this.token.text;
18417           return this.renderer.html(html);
18418         }
18419         case 'paragraph': {
18420           return this.renderer.paragraph(this.inline.output(this.token.text));
18421         }
18422         case 'text': {
18423           return this.renderer.paragraph(this.parseText());
18424         }
18425       }
18426     };
18427   
18428     
18429     /**
18430      * Marked
18431      */
18432          /**
18433          * eval:var:marked
18434     */
18435     var marked = function (src, opt, callback) {
18436       if (callback || typeof opt === 'function') {
18437         if (!callback) {
18438           callback = opt;
18439           opt = null;
18440         }
18441     
18442         opt = merge({}, marked.defaults, opt || {});
18443     
18444         var highlight = opt.highlight
18445           , tokens
18446           , pending
18447           , i = 0;
18448     
18449         try {
18450           tokens = Lexer.lex(src, opt)
18451         } catch (e) {
18452           return callback(e);
18453         }
18454     
18455         pending = tokens.length;
18456          /**
18457          * eval:var:done
18458     */
18459         var done = function(err) {
18460           if (err) {
18461             opt.highlight = highlight;
18462             return callback(err);
18463           }
18464     
18465           var out;
18466     
18467           try {
18468             out = Parser.parse(tokens, opt);
18469           } catch (e) {
18470             err = e;
18471           }
18472     
18473           opt.highlight = highlight;
18474     
18475           return err
18476             ? callback(err)
18477             : callback(null, out);
18478         };
18479     
18480         if (!highlight || highlight.length < 3) {
18481           return done();
18482         }
18483     
18484         delete opt.highlight;
18485     
18486         if (!pending) { return done(); }
18487     
18488         for (; i < tokens.length; i++) {
18489           (function(token) {
18490             if (token.type !== 'code') {
18491               return --pending || done();
18492             }
18493             return highlight(token.text, token.lang, function(err, code) {
18494               if (err) { return done(err); }
18495               if (code == null || code === token.text) {
18496                 return --pending || done();
18497               }
18498               token.text = code;
18499               token.escaped = true;
18500               --pending || done();
18501             });
18502           })(tokens[i]);
18503         }
18504     
18505         return;
18506       }
18507       try {
18508         if (opt) { opt = merge({}, marked.defaults, opt); }
18509         return Parser.parse(Lexer.lex(src, opt), opt);
18510       } catch (e) {
18511         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18512         if ((opt || marked.defaults).silent) {
18513           return '<p>An error occured:</p><pre>'
18514             + escape(e.message + '', true)
18515             + '</pre>';
18516         }
18517         throw e;
18518       }
18519     }
18520     
18521     /**
18522      * Options
18523      */
18524     
18525     marked.options =
18526     marked.setOptions = function(opt) {
18527       merge(marked.defaults, opt);
18528       return marked;
18529     };
18530     
18531     marked.defaults = {
18532       gfm: true,
18533       tables: true,
18534       breaks: false,
18535       pedantic: false,
18536       sanitize: false,
18537       sanitizer: null,
18538       mangle: true,
18539       smartLists: false,
18540       silent: false,
18541       highlight: null,
18542       langPrefix: 'lang-',
18543       smartypants: false,
18544       headerPrefix: '',
18545       renderer: new Renderer,
18546       xhtml: false
18547     };
18548     
18549     /**
18550      * Expose
18551      */
18552     
18553     marked.Parser = Parser;
18554     marked.parser = Parser.parse;
18555     
18556     marked.Renderer = Renderer;
18557     
18558     marked.Lexer = Lexer;
18559     marked.lexer = Lexer.lex;
18560     
18561     marked.InlineLexer = InlineLexer;
18562     marked.inlineLexer = InlineLexer.output;
18563     
18564     marked.parse = marked;
18565     
18566     Roo.Markdown.marked = marked;
18567
18568 })();/*
18569  * Based on:
18570  * Ext JS Library 1.1.1
18571  * Copyright(c) 2006-2007, Ext JS, LLC.
18572  *
18573  * Originally Released Under LGPL - original licence link has changed is not relivant.
18574  *
18575  * Fork - LGPL
18576  * <script type="text/javascript">
18577  */
18578
18579
18580
18581 /*
18582  * These classes are derivatives of the similarly named classes in the YUI Library.
18583  * The original license:
18584  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18585  * Code licensed under the BSD License:
18586  * http://developer.yahoo.net/yui/license.txt
18587  */
18588
18589 (function() {
18590
18591 var Event=Roo.EventManager;
18592 var Dom=Roo.lib.Dom;
18593
18594 /**
18595  * @class Roo.dd.DragDrop
18596  * @extends Roo.util.Observable
18597  * Defines the interface and base operation of items that that can be
18598  * dragged or can be drop targets.  It was designed to be extended, overriding
18599  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18600  * Up to three html elements can be associated with a DragDrop instance:
18601  * <ul>
18602  * <li>linked element: the element that is passed into the constructor.
18603  * This is the element which defines the boundaries for interaction with
18604  * other DragDrop objects.</li>
18605  * <li>handle element(s): The drag operation only occurs if the element that
18606  * was clicked matches a handle element.  By default this is the linked
18607  * element, but there are times that you will want only a portion of the
18608  * linked element to initiate the drag operation, and the setHandleElId()
18609  * method provides a way to define this.</li>
18610  * <li>drag element: this represents the element that would be moved along
18611  * with the cursor during a drag operation.  By default, this is the linked
18612  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18613  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18614  * </li>
18615  * </ul>
18616  * This class should not be instantiated until the onload event to ensure that
18617  * the associated elements are available.
18618  * The following would define a DragDrop obj that would interact with any
18619  * other DragDrop obj in the "group1" group:
18620  * <pre>
18621  *  dd = new Roo.dd.DragDrop("div1", "group1");
18622  * </pre>
18623  * Since none of the event handlers have been implemented, nothing would
18624  * actually happen if you were to run the code above.  Normally you would
18625  * override this class or one of the default implementations, but you can
18626  * also override the methods you want on an instance of the class...
18627  * <pre>
18628  *  dd.onDragDrop = function(e, id) {
18629  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18630  *  }
18631  * </pre>
18632  * @constructor
18633  * @param {String} id of the element that is linked to this instance
18634  * @param {String} sGroup the group of related DragDrop objects
18635  * @param {object} config an object containing configurable attributes
18636  *                Valid properties for DragDrop:
18637  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18638  */
18639 Roo.dd.DragDrop = function(id, sGroup, config) {
18640     if (id) {
18641         this.init(id, sGroup, config);
18642     }
18643     
18644 };
18645
18646 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18647
18648     /**
18649      * The id of the element associated with this object.  This is what we
18650      * refer to as the "linked element" because the size and position of
18651      * this element is used to determine when the drag and drop objects have
18652      * interacted.
18653      * @property id
18654      * @type String
18655      */
18656     id: null,
18657
18658     /**
18659      * Configuration attributes passed into the constructor
18660      * @property config
18661      * @type object
18662      */
18663     config: null,
18664
18665     /**
18666      * The id of the element that will be dragged.  By default this is same
18667      * as the linked element , but could be changed to another element. Ex:
18668      * Roo.dd.DDProxy
18669      * @property dragElId
18670      * @type String
18671      * @private
18672      */
18673     dragElId: null,
18674
18675     /**
18676      * the id of the element that initiates the drag operation.  By default
18677      * this is the linked element, but could be changed to be a child of this
18678      * element.  This lets us do things like only starting the drag when the
18679      * header element within the linked html element is clicked.
18680      * @property handleElId
18681      * @type String
18682      * @private
18683      */
18684     handleElId: null,
18685
18686     /**
18687      * An associative array of HTML tags that will be ignored if clicked.
18688      * @property invalidHandleTypes
18689      * @type {string: string}
18690      */
18691     invalidHandleTypes: null,
18692
18693     /**
18694      * An associative array of ids for elements that will be ignored if clicked
18695      * @property invalidHandleIds
18696      * @type {string: string}
18697      */
18698     invalidHandleIds: null,
18699
18700     /**
18701      * An indexted array of css class names for elements that will be ignored
18702      * if clicked.
18703      * @property invalidHandleClasses
18704      * @type string[]
18705      */
18706     invalidHandleClasses: null,
18707
18708     /**
18709      * The linked element's absolute X position at the time the drag was
18710      * started
18711      * @property startPageX
18712      * @type int
18713      * @private
18714      */
18715     startPageX: 0,
18716
18717     /**
18718      * The linked element's absolute X position at the time the drag was
18719      * started
18720      * @property startPageY
18721      * @type int
18722      * @private
18723      */
18724     startPageY: 0,
18725
18726     /**
18727      * The group defines a logical collection of DragDrop objects that are
18728      * related.  Instances only get events when interacting with other
18729      * DragDrop object in the same group.  This lets us define multiple
18730      * groups using a single DragDrop subclass if we want.
18731      * @property groups
18732      * @type {string: string}
18733      */
18734     groups: null,
18735
18736     /**
18737      * Individual drag/drop instances can be locked.  This will prevent
18738      * onmousedown start drag.
18739      * @property locked
18740      * @type boolean
18741      * @private
18742      */
18743     locked: false,
18744
18745     /**
18746      * Lock this instance
18747      * @method lock
18748      */
18749     lock: function() { this.locked = true; },
18750
18751     /**
18752      * Unlock this instace
18753      * @method unlock
18754      */
18755     unlock: function() { this.locked = false; },
18756
18757     /**
18758      * By default, all insances can be a drop target.  This can be disabled by
18759      * setting isTarget to false.
18760      * @method isTarget
18761      * @type boolean
18762      */
18763     isTarget: true,
18764
18765     /**
18766      * The padding configured for this drag and drop object for calculating
18767      * the drop zone intersection with this object.
18768      * @method padding
18769      * @type int[]
18770      */
18771     padding: null,
18772
18773     /**
18774      * Cached reference to the linked element
18775      * @property _domRef
18776      * @private
18777      */
18778     _domRef: null,
18779
18780     /**
18781      * Internal typeof flag
18782      * @property __ygDragDrop
18783      * @private
18784      */
18785     __ygDragDrop: true,
18786
18787     /**
18788      * Set to true when horizontal contraints are applied
18789      * @property constrainX
18790      * @type boolean
18791      * @private
18792      */
18793     constrainX: false,
18794
18795     /**
18796      * Set to true when vertical contraints are applied
18797      * @property constrainY
18798      * @type boolean
18799      * @private
18800      */
18801     constrainY: false,
18802
18803     /**
18804      * The left constraint
18805      * @property minX
18806      * @type int
18807      * @private
18808      */
18809     minX: 0,
18810
18811     /**
18812      * The right constraint
18813      * @property maxX
18814      * @type int
18815      * @private
18816      */
18817     maxX: 0,
18818
18819     /**
18820      * The up constraint
18821      * @property minY
18822      * @type int
18823      * @type int
18824      * @private
18825      */
18826     minY: 0,
18827
18828     /**
18829      * The down constraint
18830      * @property maxY
18831      * @type int
18832      * @private
18833      */
18834     maxY: 0,
18835
18836     /**
18837      * Maintain offsets when we resetconstraints.  Set to true when you want
18838      * the position of the element relative to its parent to stay the same
18839      * when the page changes
18840      *
18841      * @property maintainOffset
18842      * @type boolean
18843      */
18844     maintainOffset: false,
18845
18846     /**
18847      * Array of pixel locations the element will snap to if we specified a
18848      * horizontal graduation/interval.  This array is generated automatically
18849      * when you define a tick interval.
18850      * @property xTicks
18851      * @type int[]
18852      */
18853     xTicks: null,
18854
18855     /**
18856      * Array of pixel locations the element will snap to if we specified a
18857      * vertical graduation/interval.  This array is generated automatically
18858      * when you define a tick interval.
18859      * @property yTicks
18860      * @type int[]
18861      */
18862     yTicks: null,
18863
18864     /**
18865      * By default the drag and drop instance will only respond to the primary
18866      * button click (left button for a right-handed mouse).  Set to true to
18867      * allow drag and drop to start with any mouse click that is propogated
18868      * by the browser
18869      * @property primaryButtonOnly
18870      * @type boolean
18871      */
18872     primaryButtonOnly: true,
18873
18874     /**
18875      * The availabe property is false until the linked dom element is accessible.
18876      * @property available
18877      * @type boolean
18878      */
18879     available: false,
18880
18881     /**
18882      * By default, drags can only be initiated if the mousedown occurs in the
18883      * region the linked element is.  This is done in part to work around a
18884      * bug in some browsers that mis-report the mousedown if the previous
18885      * mouseup happened outside of the window.  This property is set to true
18886      * if outer handles are defined.
18887      *
18888      * @property hasOuterHandles
18889      * @type boolean
18890      * @default false
18891      */
18892     hasOuterHandles: false,
18893
18894     /**
18895      * Code that executes immediately before the startDrag event
18896      * @method b4StartDrag
18897      * @private
18898      */
18899     b4StartDrag: function(x, y) { },
18900
18901     /**
18902      * Abstract method called after a drag/drop object is clicked
18903      * and the drag or mousedown time thresholds have beeen met.
18904      * @method startDrag
18905      * @param {int} X click location
18906      * @param {int} Y click location
18907      */
18908     startDrag: function(x, y) { /* override this */ },
18909
18910     /**
18911      * Code that executes immediately before the onDrag event
18912      * @method b4Drag
18913      * @private
18914      */
18915     b4Drag: function(e) { },
18916
18917     /**
18918      * Abstract method called during the onMouseMove event while dragging an
18919      * object.
18920      * @method onDrag
18921      * @param {Event} e the mousemove event
18922      */
18923     onDrag: function(e) { /* override this */ },
18924
18925     /**
18926      * Abstract method called when this element fist begins hovering over
18927      * another DragDrop obj
18928      * @method onDragEnter
18929      * @param {Event} e the mousemove event
18930      * @param {String|DragDrop[]} id In POINT mode, the element
18931      * id this is hovering over.  In INTERSECT mode, an array of one or more
18932      * dragdrop items being hovered over.
18933      */
18934     onDragEnter: function(e, id) { /* override this */ },
18935
18936     /**
18937      * Code that executes immediately before the onDragOver event
18938      * @method b4DragOver
18939      * @private
18940      */
18941     b4DragOver: function(e) { },
18942
18943     /**
18944      * Abstract method called when this element is hovering over another
18945      * DragDrop obj
18946      * @method onDragOver
18947      * @param {Event} e the mousemove event
18948      * @param {String|DragDrop[]} id In POINT mode, the element
18949      * id this is hovering over.  In INTERSECT mode, an array of dd items
18950      * being hovered over.
18951      */
18952     onDragOver: function(e, id) { /* override this */ },
18953
18954     /**
18955      * Code that executes immediately before the onDragOut event
18956      * @method b4DragOut
18957      * @private
18958      */
18959     b4DragOut: function(e) { },
18960
18961     /**
18962      * Abstract method called when we are no longer hovering over an element
18963      * @method onDragOut
18964      * @param {Event} e the mousemove event
18965      * @param {String|DragDrop[]} id In POINT mode, the element
18966      * id this was hovering over.  In INTERSECT mode, an array of dd items
18967      * that the mouse is no longer over.
18968      */
18969     onDragOut: function(e, id) { /* override this */ },
18970
18971     /**
18972      * Code that executes immediately before the onDragDrop event
18973      * @method b4DragDrop
18974      * @private
18975      */
18976     b4DragDrop: function(e) { },
18977
18978     /**
18979      * Abstract method called when this item is dropped on another DragDrop
18980      * obj
18981      * @method onDragDrop
18982      * @param {Event} e the mouseup event
18983      * @param {String|DragDrop[]} id In POINT mode, the element
18984      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18985      * was dropped on.
18986      */
18987     onDragDrop: function(e, id) { /* override this */ },
18988
18989     /**
18990      * Abstract method called when this item is dropped on an area with no
18991      * drop target
18992      * @method onInvalidDrop
18993      * @param {Event} e the mouseup event
18994      */
18995     onInvalidDrop: function(e) { /* override this */ },
18996
18997     /**
18998      * Code that executes immediately before the endDrag event
18999      * @method b4EndDrag
19000      * @private
19001      */
19002     b4EndDrag: function(e) { },
19003
19004     /**
19005      * Fired when we are done dragging the object
19006      * @method endDrag
19007      * @param {Event} e the mouseup event
19008      */
19009     endDrag: function(e) { /* override this */ },
19010
19011     /**
19012      * Code executed immediately before the onMouseDown event
19013      * @method b4MouseDown
19014      * @param {Event} e the mousedown event
19015      * @private
19016      */
19017     b4MouseDown: function(e) {  },
19018
19019     /**
19020      * Event handler that fires when a drag/drop obj gets a mousedown
19021      * @method onMouseDown
19022      * @param {Event} e the mousedown event
19023      */
19024     onMouseDown: function(e) { /* override this */ },
19025
19026     /**
19027      * Event handler that fires when a drag/drop obj gets a mouseup
19028      * @method onMouseUp
19029      * @param {Event} e the mouseup event
19030      */
19031     onMouseUp: function(e) { /* override this */ },
19032
19033     /**
19034      * Override the onAvailable method to do what is needed after the initial
19035      * position was determined.
19036      * @method onAvailable
19037      */
19038     onAvailable: function () {
19039     },
19040
19041     /*
19042      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
19043      * @type Object
19044      */
19045     defaultPadding : {left:0, right:0, top:0, bottom:0},
19046
19047     /*
19048      * Initializes the drag drop object's constraints to restrict movement to a certain element.
19049  *
19050  * Usage:
19051  <pre><code>
19052  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
19053                 { dragElId: "existingProxyDiv" });
19054  dd.startDrag = function(){
19055      this.constrainTo("parent-id");
19056  };
19057  </code></pre>
19058  * Or you can initalize it using the {@link Roo.Element} object:
19059  <pre><code>
19060  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
19061      startDrag : function(){
19062          this.constrainTo("parent-id");
19063      }
19064  });
19065  </code></pre>
19066      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
19067      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
19068      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
19069      * an object containing the sides to pad. For example: {right:10, bottom:10}
19070      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
19071      */
19072     constrainTo : function(constrainTo, pad, inContent){
19073         if(typeof pad == "number"){
19074             pad = {left: pad, right:pad, top:pad, bottom:pad};
19075         }
19076         pad = pad || this.defaultPadding;
19077         var b = Roo.get(this.getEl()).getBox();
19078         var ce = Roo.get(constrainTo);
19079         var s = ce.getScroll();
19080         var c, cd = ce.dom;
19081         if(cd == document.body){
19082             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
19083         }else{
19084             xy = ce.getXY();
19085             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
19086         }
19087
19088
19089         var topSpace = b.y - c.y;
19090         var leftSpace = b.x - c.x;
19091
19092         this.resetConstraints();
19093         this.setXConstraint(leftSpace - (pad.left||0), // left
19094                 c.width - leftSpace - b.width - (pad.right||0) //right
19095         );
19096         this.setYConstraint(topSpace - (pad.top||0), //top
19097                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
19098         );
19099     },
19100
19101     /**
19102      * Returns a reference to the linked element
19103      * @method getEl
19104      * @return {HTMLElement} the html element
19105      */
19106     getEl: function() {
19107         if (!this._domRef) {
19108             this._domRef = Roo.getDom(this.id);
19109         }
19110
19111         return this._domRef;
19112     },
19113
19114     /**
19115      * Returns a reference to the actual element to drag.  By default this is
19116      * the same as the html element, but it can be assigned to another
19117      * element. An example of this can be found in Roo.dd.DDProxy
19118      * @method getDragEl
19119      * @return {HTMLElement} the html element
19120      */
19121     getDragEl: function() {
19122         return Roo.getDom(this.dragElId);
19123     },
19124
19125     /**
19126      * Sets up the DragDrop object.  Must be called in the constructor of any
19127      * Roo.dd.DragDrop subclass
19128      * @method init
19129      * @param id the id of the linked element
19130      * @param {String} sGroup the group of related items
19131      * @param {object} config configuration attributes
19132      */
19133     init: function(id, sGroup, config) {
19134         this.initTarget(id, sGroup, config);
19135         if (!Roo.isTouch) {
19136             Event.on(this.id, "mousedown", this.handleMouseDown, this);
19137         }
19138         Event.on(this.id, "touchstart", this.handleMouseDown, this);
19139         // Event.on(this.id, "selectstart", Event.preventDefault);
19140     },
19141
19142     /**
19143      * Initializes Targeting functionality only... the object does not
19144      * get a mousedown handler.
19145      * @method initTarget
19146      * @param id the id of the linked element
19147      * @param {String} sGroup the group of related items
19148      * @param {object} config configuration attributes
19149      */
19150     initTarget: function(id, sGroup, config) {
19151
19152         // configuration attributes
19153         this.config = config || {};
19154
19155         // create a local reference to the drag and drop manager
19156         this.DDM = Roo.dd.DDM;
19157         // initialize the groups array
19158         this.groups = {};
19159
19160         // assume that we have an element reference instead of an id if the
19161         // parameter is not a string
19162         if (typeof id !== "string") {
19163             id = Roo.id(id);
19164         }
19165
19166         // set the id
19167         this.id = id;
19168
19169         // add to an interaction group
19170         this.addToGroup((sGroup) ? sGroup : "default");
19171
19172         // We don't want to register this as the handle with the manager
19173         // so we just set the id rather than calling the setter.
19174         this.handleElId = id;
19175
19176         // the linked element is the element that gets dragged by default
19177         this.setDragElId(id);
19178
19179         // by default, clicked anchors will not start drag operations.
19180         this.invalidHandleTypes = { A: "A" };
19181         this.invalidHandleIds = {};
19182         this.invalidHandleClasses = [];
19183
19184         this.applyConfig();
19185
19186         this.handleOnAvailable();
19187     },
19188
19189     /**
19190      * Applies the configuration parameters that were passed into the constructor.
19191      * This is supposed to happen at each level through the inheritance chain.  So
19192      * a DDProxy implentation will execute apply config on DDProxy, DD, and
19193      * DragDrop in order to get all of the parameters that are available in
19194      * each object.
19195      * @method applyConfig
19196      */
19197     applyConfig: function() {
19198
19199         // configurable properties:
19200         //    padding, isTarget, maintainOffset, primaryButtonOnly
19201         this.padding           = this.config.padding || [0, 0, 0, 0];
19202         this.isTarget          = (this.config.isTarget !== false);
19203         this.maintainOffset    = (this.config.maintainOffset);
19204         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
19205
19206     },
19207
19208     /**
19209      * Executed when the linked element is available
19210      * @method handleOnAvailable
19211      * @private
19212      */
19213     handleOnAvailable: function() {
19214         this.available = true;
19215         this.resetConstraints();
19216         this.onAvailable();
19217     },
19218
19219      /**
19220      * Configures the padding for the target zone in px.  Effectively expands
19221      * (or reduces) the virtual object size for targeting calculations.
19222      * Supports css-style shorthand; if only one parameter is passed, all sides
19223      * will have that padding, and if only two are passed, the top and bottom
19224      * will have the first param, the left and right the second.
19225      * @method setPadding
19226      * @param {int} iTop    Top pad
19227      * @param {int} iRight  Right pad
19228      * @param {int} iBot    Bot pad
19229      * @param {int} iLeft   Left pad
19230      */
19231     setPadding: function(iTop, iRight, iBot, iLeft) {
19232         // this.padding = [iLeft, iRight, iTop, iBot];
19233         if (!iRight && 0 !== iRight) {
19234             this.padding = [iTop, iTop, iTop, iTop];
19235         } else if (!iBot && 0 !== iBot) {
19236             this.padding = [iTop, iRight, iTop, iRight];
19237         } else {
19238             this.padding = [iTop, iRight, iBot, iLeft];
19239         }
19240     },
19241
19242     /**
19243      * Stores the initial placement of the linked element.
19244      * @method setInitialPosition
19245      * @param {int} diffX   the X offset, default 0
19246      * @param {int} diffY   the Y offset, default 0
19247      */
19248     setInitPosition: function(diffX, diffY) {
19249         var el = this.getEl();
19250
19251         if (!this.DDM.verifyEl(el)) {
19252             return;
19253         }
19254
19255         var dx = diffX || 0;
19256         var dy = diffY || 0;
19257
19258         var p = Dom.getXY( el );
19259
19260         this.initPageX = p[0] - dx;
19261         this.initPageY = p[1] - dy;
19262
19263         this.lastPageX = p[0];
19264         this.lastPageY = p[1];
19265
19266
19267         this.setStartPosition(p);
19268     },
19269
19270     /**
19271      * Sets the start position of the element.  This is set when the obj
19272      * is initialized, the reset when a drag is started.
19273      * @method setStartPosition
19274      * @param pos current position (from previous lookup)
19275      * @private
19276      */
19277     setStartPosition: function(pos) {
19278         var p = pos || Dom.getXY( this.getEl() );
19279         this.deltaSetXY = null;
19280
19281         this.startPageX = p[0];
19282         this.startPageY = p[1];
19283     },
19284
19285     /**
19286      * Add this instance to a group of related drag/drop objects.  All
19287      * instances belong to at least one group, and can belong to as many
19288      * groups as needed.
19289      * @method addToGroup
19290      * @param sGroup {string} the name of the group
19291      */
19292     addToGroup: function(sGroup) {
19293         this.groups[sGroup] = true;
19294         this.DDM.regDragDrop(this, sGroup);
19295     },
19296
19297     /**
19298      * Remove's this instance from the supplied interaction group
19299      * @method removeFromGroup
19300      * @param {string}  sGroup  The group to drop
19301      */
19302     removeFromGroup: function(sGroup) {
19303         if (this.groups[sGroup]) {
19304             delete this.groups[sGroup];
19305         }
19306
19307         this.DDM.removeDDFromGroup(this, sGroup);
19308     },
19309
19310     /**
19311      * Allows you to specify that an element other than the linked element
19312      * will be moved with the cursor during a drag
19313      * @method setDragElId
19314      * @param id {string} the id of the element that will be used to initiate the drag
19315      */
19316     setDragElId: function(id) {
19317         this.dragElId = id;
19318     },
19319
19320     /**
19321      * Allows you to specify a child of the linked element that should be
19322      * used to initiate the drag operation.  An example of this would be if
19323      * you have a content div with text and links.  Clicking anywhere in the
19324      * content area would normally start the drag operation.  Use this method
19325      * to specify that an element inside of the content div is the element
19326      * that starts the drag operation.
19327      * @method setHandleElId
19328      * @param id {string} the id of the element that will be used to
19329      * initiate the drag.
19330      */
19331     setHandleElId: function(id) {
19332         if (typeof id !== "string") {
19333             id = Roo.id(id);
19334         }
19335         this.handleElId = id;
19336         this.DDM.regHandle(this.id, id);
19337     },
19338
19339     /**
19340      * Allows you to set an element outside of the linked element as a drag
19341      * handle
19342      * @method setOuterHandleElId
19343      * @param id the id of the element that will be used to initiate the drag
19344      */
19345     setOuterHandleElId: function(id) {
19346         if (typeof id !== "string") {
19347             id = Roo.id(id);
19348         }
19349         Event.on(id, "mousedown",
19350                 this.handleMouseDown, this);
19351         this.setHandleElId(id);
19352
19353         this.hasOuterHandles = true;
19354     },
19355
19356     /**
19357      * Remove all drag and drop hooks for this element
19358      * @method unreg
19359      */
19360     unreg: function() {
19361         Event.un(this.id, "mousedown",
19362                 this.handleMouseDown);
19363         Event.un(this.id, "touchstart",
19364                 this.handleMouseDown);
19365         this._domRef = null;
19366         this.DDM._remove(this);
19367     },
19368
19369     destroy : function(){
19370         this.unreg();
19371     },
19372
19373     /**
19374      * Returns true if this instance is locked, or the drag drop mgr is locked
19375      * (meaning that all drag/drop is disabled on the page.)
19376      * @method isLocked
19377      * @return {boolean} true if this obj or all drag/drop is locked, else
19378      * false
19379      */
19380     isLocked: function() {
19381         return (this.DDM.isLocked() || this.locked);
19382     },
19383
19384     /**
19385      * Fired when this object is clicked
19386      * @method handleMouseDown
19387      * @param {Event} e
19388      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
19389      * @private
19390      */
19391     handleMouseDown: function(e, oDD){
19392      
19393         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
19394             //Roo.log('not touch/ button !=0');
19395             return;
19396         }
19397         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
19398             return; // double touch..
19399         }
19400         
19401
19402         if (this.isLocked()) {
19403             //Roo.log('locked');
19404             return;
19405         }
19406
19407         this.DDM.refreshCache(this.groups);
19408 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
19409         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
19410         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
19411             //Roo.log('no outer handes or not over target');
19412                 // do nothing.
19413         } else {
19414 //            Roo.log('check validator');
19415             if (this.clickValidator(e)) {
19416 //                Roo.log('validate success');
19417                 // set the initial element position
19418                 this.setStartPosition();
19419
19420
19421                 this.b4MouseDown(e);
19422                 this.onMouseDown(e);
19423
19424                 this.DDM.handleMouseDown(e, this);
19425
19426                 this.DDM.stopEvent(e);
19427             } else {
19428
19429
19430             }
19431         }
19432     },
19433
19434     clickValidator: function(e) {
19435         var target = e.getTarget();
19436         return ( this.isValidHandleChild(target) &&
19437                     (this.id == this.handleElId ||
19438                         this.DDM.handleWasClicked(target, this.id)) );
19439     },
19440
19441     /**
19442      * Allows you to specify a tag name that should not start a drag operation
19443      * when clicked.  This is designed to facilitate embedding links within a
19444      * drag handle that do something other than start the drag.
19445      * @method addInvalidHandleType
19446      * @param {string} tagName the type of element to exclude
19447      */
19448     addInvalidHandleType: function(tagName) {
19449         var type = tagName.toUpperCase();
19450         this.invalidHandleTypes[type] = type;
19451     },
19452
19453     /**
19454      * Lets you to specify an element id for a child of a drag handle
19455      * that should not initiate a drag
19456      * @method addInvalidHandleId
19457      * @param {string} id the element id of the element you wish to ignore
19458      */
19459     addInvalidHandleId: function(id) {
19460         if (typeof id !== "string") {
19461             id = Roo.id(id);
19462         }
19463         this.invalidHandleIds[id] = id;
19464     },
19465
19466     /**
19467      * Lets you specify a css class of elements that will not initiate a drag
19468      * @method addInvalidHandleClass
19469      * @param {string} cssClass the class of the elements you wish to ignore
19470      */
19471     addInvalidHandleClass: function(cssClass) {
19472         this.invalidHandleClasses.push(cssClass);
19473     },
19474
19475     /**
19476      * Unsets an excluded tag name set by addInvalidHandleType
19477      * @method removeInvalidHandleType
19478      * @param {string} tagName the type of element to unexclude
19479      */
19480     removeInvalidHandleType: function(tagName) {
19481         var type = tagName.toUpperCase();
19482         // this.invalidHandleTypes[type] = null;
19483         delete this.invalidHandleTypes[type];
19484     },
19485
19486     /**
19487      * Unsets an invalid handle id
19488      * @method removeInvalidHandleId
19489      * @param {string} id the id of the element to re-enable
19490      */
19491     removeInvalidHandleId: function(id) {
19492         if (typeof id !== "string") {
19493             id = Roo.id(id);
19494         }
19495         delete this.invalidHandleIds[id];
19496     },
19497
19498     /**
19499      * Unsets an invalid css class
19500      * @method removeInvalidHandleClass
19501      * @param {string} cssClass the class of the element(s) you wish to
19502      * re-enable
19503      */
19504     removeInvalidHandleClass: function(cssClass) {
19505         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19506             if (this.invalidHandleClasses[i] == cssClass) {
19507                 delete this.invalidHandleClasses[i];
19508             }
19509         }
19510     },
19511
19512     /**
19513      * Checks the tag exclusion list to see if this click should be ignored
19514      * @method isValidHandleChild
19515      * @param {HTMLElement} node the HTMLElement to evaluate
19516      * @return {boolean} true if this is a valid tag type, false if not
19517      */
19518     isValidHandleChild: function(node) {
19519
19520         var valid = true;
19521         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19522         var nodeName;
19523         try {
19524             nodeName = node.nodeName.toUpperCase();
19525         } catch(e) {
19526             nodeName = node.nodeName;
19527         }
19528         valid = valid && !this.invalidHandleTypes[nodeName];
19529         valid = valid && !this.invalidHandleIds[node.id];
19530
19531         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19532             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19533         }
19534
19535
19536         return valid;
19537
19538     },
19539
19540     /**
19541      * Create the array of horizontal tick marks if an interval was specified
19542      * in setXConstraint().
19543      * @method setXTicks
19544      * @private
19545      */
19546     setXTicks: function(iStartX, iTickSize) {
19547         this.xTicks = [];
19548         this.xTickSize = iTickSize;
19549
19550         var tickMap = {};
19551
19552         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19553             if (!tickMap[i]) {
19554                 this.xTicks[this.xTicks.length] = i;
19555                 tickMap[i] = true;
19556             }
19557         }
19558
19559         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19560             if (!tickMap[i]) {
19561                 this.xTicks[this.xTicks.length] = i;
19562                 tickMap[i] = true;
19563             }
19564         }
19565
19566         this.xTicks.sort(this.DDM.numericSort) ;
19567     },
19568
19569     /**
19570      * Create the array of vertical tick marks if an interval was specified in
19571      * setYConstraint().
19572      * @method setYTicks
19573      * @private
19574      */
19575     setYTicks: function(iStartY, iTickSize) {
19576         this.yTicks = [];
19577         this.yTickSize = iTickSize;
19578
19579         var tickMap = {};
19580
19581         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19582             if (!tickMap[i]) {
19583                 this.yTicks[this.yTicks.length] = i;
19584                 tickMap[i] = true;
19585             }
19586         }
19587
19588         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19589             if (!tickMap[i]) {
19590                 this.yTicks[this.yTicks.length] = i;
19591                 tickMap[i] = true;
19592             }
19593         }
19594
19595         this.yTicks.sort(this.DDM.numericSort) ;
19596     },
19597
19598     /**
19599      * By default, the element can be dragged any place on the screen.  Use
19600      * this method to limit the horizontal travel of the element.  Pass in
19601      * 0,0 for the parameters if you want to lock the drag to the y axis.
19602      * @method setXConstraint
19603      * @param {int} iLeft the number of pixels the element can move to the left
19604      * @param {int} iRight the number of pixels the element can move to the
19605      * right
19606      * @param {int} iTickSize optional parameter for specifying that the
19607      * element
19608      * should move iTickSize pixels at a time.
19609      */
19610     setXConstraint: function(iLeft, iRight, iTickSize) {
19611         this.leftConstraint = iLeft;
19612         this.rightConstraint = iRight;
19613
19614         this.minX = this.initPageX - iLeft;
19615         this.maxX = this.initPageX + iRight;
19616         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19617
19618         this.constrainX = true;
19619     },
19620
19621     /**
19622      * Clears any constraints applied to this instance.  Also clears ticks
19623      * since they can't exist independent of a constraint at this time.
19624      * @method clearConstraints
19625      */
19626     clearConstraints: function() {
19627         this.constrainX = false;
19628         this.constrainY = false;
19629         this.clearTicks();
19630     },
19631
19632     /**
19633      * Clears any tick interval defined for this instance
19634      * @method clearTicks
19635      */
19636     clearTicks: function() {
19637         this.xTicks = null;
19638         this.yTicks = null;
19639         this.xTickSize = 0;
19640         this.yTickSize = 0;
19641     },
19642
19643     /**
19644      * By default, the element can be dragged any place on the screen.  Set
19645      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19646      * parameters if you want to lock the drag to the x axis.
19647      * @method setYConstraint
19648      * @param {int} iUp the number of pixels the element can move up
19649      * @param {int} iDown the number of pixels the element can move down
19650      * @param {int} iTickSize optional parameter for specifying that the
19651      * element should move iTickSize pixels at a time.
19652      */
19653     setYConstraint: function(iUp, iDown, iTickSize) {
19654         this.topConstraint = iUp;
19655         this.bottomConstraint = iDown;
19656
19657         this.minY = this.initPageY - iUp;
19658         this.maxY = this.initPageY + iDown;
19659         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19660
19661         this.constrainY = true;
19662
19663     },
19664
19665     /**
19666      * resetConstraints must be called if you manually reposition a dd element.
19667      * @method resetConstraints
19668      * @param {boolean} maintainOffset
19669      */
19670     resetConstraints: function() {
19671
19672
19673         // Maintain offsets if necessary
19674         if (this.initPageX || this.initPageX === 0) {
19675             // figure out how much this thing has moved
19676             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19677             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19678
19679             this.setInitPosition(dx, dy);
19680
19681         // This is the first time we have detected the element's position
19682         } else {
19683             this.setInitPosition();
19684         }
19685
19686         if (this.constrainX) {
19687             this.setXConstraint( this.leftConstraint,
19688                                  this.rightConstraint,
19689                                  this.xTickSize        );
19690         }
19691
19692         if (this.constrainY) {
19693             this.setYConstraint( this.topConstraint,
19694                                  this.bottomConstraint,
19695                                  this.yTickSize         );
19696         }
19697     },
19698
19699     /**
19700      * Normally the drag element is moved pixel by pixel, but we can specify
19701      * that it move a number of pixels at a time.  This method resolves the
19702      * location when we have it set up like this.
19703      * @method getTick
19704      * @param {int} val where we want to place the object
19705      * @param {int[]} tickArray sorted array of valid points
19706      * @return {int} the closest tick
19707      * @private
19708      */
19709     getTick: function(val, tickArray) {
19710
19711         if (!tickArray) {
19712             // If tick interval is not defined, it is effectively 1 pixel,
19713             // so we return the value passed to us.
19714             return val;
19715         } else if (tickArray[0] >= val) {
19716             // The value is lower than the first tick, so we return the first
19717             // tick.
19718             return tickArray[0];
19719         } else {
19720             for (var i=0, len=tickArray.length; i<len; ++i) {
19721                 var next = i + 1;
19722                 if (tickArray[next] && tickArray[next] >= val) {
19723                     var diff1 = val - tickArray[i];
19724                     var diff2 = tickArray[next] - val;
19725                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19726                 }
19727             }
19728
19729             // The value is larger than the last tick, so we return the last
19730             // tick.
19731             return tickArray[tickArray.length - 1];
19732         }
19733     },
19734
19735     /**
19736      * toString method
19737      * @method toString
19738      * @return {string} string representation of the dd obj
19739      */
19740     toString: function() {
19741         return ("DragDrop " + this.id);
19742     }
19743
19744 });
19745
19746 })();
19747 /*
19748  * Based on:
19749  * Ext JS Library 1.1.1
19750  * Copyright(c) 2006-2007, Ext JS, LLC.
19751  *
19752  * Originally Released Under LGPL - original licence link has changed is not relivant.
19753  *
19754  * Fork - LGPL
19755  * <script type="text/javascript">
19756  */
19757
19758
19759 /**
19760  * The drag and drop utility provides a framework for building drag and drop
19761  * applications.  In addition to enabling drag and drop for specific elements,
19762  * the drag and drop elements are tracked by the manager class, and the
19763  * interactions between the various elements are tracked during the drag and
19764  * the implementing code is notified about these important moments.
19765  */
19766
19767 // Only load the library once.  Rewriting the manager class would orphan
19768 // existing drag and drop instances.
19769 if (!Roo.dd.DragDropMgr) {
19770
19771 /**
19772  * @class Roo.dd.DragDropMgr
19773  * DragDropMgr is a singleton that tracks the element interaction for
19774  * all DragDrop items in the window.  Generally, you will not call
19775  * this class directly, but it does have helper methods that could
19776  * be useful in your DragDrop implementations.
19777  * @singleton
19778  */
19779 Roo.dd.DragDropMgr = function() {
19780
19781     var Event = Roo.EventManager;
19782
19783     return {
19784
19785         /**
19786          * Two dimensional Array of registered DragDrop objects.  The first
19787          * dimension is the DragDrop item group, the second the DragDrop
19788          * object.
19789          * @property ids
19790          * @type {string: string}
19791          * @private
19792          * @static
19793          */
19794         ids: {},
19795
19796         /**
19797          * Array of element ids defined as drag handles.  Used to determine
19798          * if the element that generated the mousedown event is actually the
19799          * handle and not the html element itself.
19800          * @property handleIds
19801          * @type {string: string}
19802          * @private
19803          * @static
19804          */
19805         handleIds: {},
19806
19807         /**
19808          * the DragDrop object that is currently being dragged
19809          * @property dragCurrent
19810          * @type DragDrop
19811          * @private
19812          * @static
19813          **/
19814         dragCurrent: null,
19815
19816         /**
19817          * the DragDrop object(s) that are being hovered over
19818          * @property dragOvers
19819          * @type Array
19820          * @private
19821          * @static
19822          */
19823         dragOvers: {},
19824
19825         /**
19826          * the X distance between the cursor and the object being dragged
19827          * @property deltaX
19828          * @type int
19829          * @private
19830          * @static
19831          */
19832         deltaX: 0,
19833
19834         /**
19835          * the Y distance between the cursor and the object being dragged
19836          * @property deltaY
19837          * @type int
19838          * @private
19839          * @static
19840          */
19841         deltaY: 0,
19842
19843         /**
19844          * Flag to determine if we should prevent the default behavior of the
19845          * events we define. By default this is true, but this can be set to
19846          * false if you need the default behavior (not recommended)
19847          * @property preventDefault
19848          * @type boolean
19849          * @static
19850          */
19851         preventDefault: true,
19852
19853         /**
19854          * Flag to determine if we should stop the propagation of the events
19855          * we generate. This is true by default but you may want to set it to
19856          * false if the html element contains other features that require the
19857          * mouse click.
19858          * @property stopPropagation
19859          * @type boolean
19860          * @static
19861          */
19862         stopPropagation: true,
19863
19864         /**
19865          * Internal flag that is set to true when drag and drop has been
19866          * intialized
19867          * @property initialized
19868          * @private
19869          * @static
19870          */
19871         initalized: false,
19872
19873         /**
19874          * All drag and drop can be disabled.
19875          * @property locked
19876          * @private
19877          * @static
19878          */
19879         locked: false,
19880
19881         /**
19882          * Called the first time an element is registered.
19883          * @method init
19884          * @private
19885          * @static
19886          */
19887         init: function() {
19888             this.initialized = true;
19889         },
19890
19891         /**
19892          * In point mode, drag and drop interaction is defined by the
19893          * location of the cursor during the drag/drop
19894          * @property POINT
19895          * @type int
19896          * @static
19897          */
19898         POINT: 0,
19899
19900         /**
19901          * In intersect mode, drag and drop interactio nis defined by the
19902          * overlap of two or more drag and drop objects.
19903          * @property INTERSECT
19904          * @type int
19905          * @static
19906          */
19907         INTERSECT: 1,
19908
19909         /**
19910          * The current drag and drop mode.  Default: POINT
19911          * @property mode
19912          * @type int
19913          * @static
19914          */
19915         mode: 0,
19916
19917         /**
19918          * Runs method on all drag and drop objects
19919          * @method _execOnAll
19920          * @private
19921          * @static
19922          */
19923         _execOnAll: function(sMethod, args) {
19924             for (var i in this.ids) {
19925                 for (var j in this.ids[i]) {
19926                     var oDD = this.ids[i][j];
19927                     if (! this.isTypeOfDD(oDD)) {
19928                         continue;
19929                     }
19930                     oDD[sMethod].apply(oDD, args);
19931                 }
19932             }
19933         },
19934
19935         /**
19936          * Drag and drop initialization.  Sets up the global event handlers
19937          * @method _onLoad
19938          * @private
19939          * @static
19940          */
19941         _onLoad: function() {
19942
19943             this.init();
19944
19945             if (!Roo.isTouch) {
19946                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19947                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19948             }
19949             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19950             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19951             
19952             Event.on(window,   "unload",    this._onUnload, this, true);
19953             Event.on(window,   "resize",    this._onResize, this, true);
19954             // Event.on(window,   "mouseout",    this._test);
19955
19956         },
19957
19958         /**
19959          * Reset constraints on all drag and drop objs
19960          * @method _onResize
19961          * @private
19962          * @static
19963          */
19964         _onResize: function(e) {
19965             this._execOnAll("resetConstraints", []);
19966         },
19967
19968         /**
19969          * Lock all drag and drop functionality
19970          * @method lock
19971          * @static
19972          */
19973         lock: function() { this.locked = true; },
19974
19975         /**
19976          * Unlock all drag and drop functionality
19977          * @method unlock
19978          * @static
19979          */
19980         unlock: function() { this.locked = false; },
19981
19982         /**
19983          * Is drag and drop locked?
19984          * @method isLocked
19985          * @return {boolean} True if drag and drop is locked, false otherwise.
19986          * @static
19987          */
19988         isLocked: function() { return this.locked; },
19989
19990         /**
19991          * Location cache that is set for all drag drop objects when a drag is
19992          * initiated, cleared when the drag is finished.
19993          * @property locationCache
19994          * @private
19995          * @static
19996          */
19997         locationCache: {},
19998
19999         /**
20000          * Set useCache to false if you want to force object the lookup of each
20001          * drag and drop linked element constantly during a drag.
20002          * @property useCache
20003          * @type boolean
20004          * @static
20005          */
20006         useCache: true,
20007
20008         /**
20009          * The number of pixels that the mouse needs to move after the
20010          * mousedown before the drag is initiated.  Default=3;
20011          * @property clickPixelThresh
20012          * @type int
20013          * @static
20014          */
20015         clickPixelThresh: 3,
20016
20017         /**
20018          * The number of milliseconds after the mousedown event to initiate the
20019          * drag if we don't get a mouseup event. Default=1000
20020          * @property clickTimeThresh
20021          * @type int
20022          * @static
20023          */
20024         clickTimeThresh: 350,
20025
20026         /**
20027          * Flag that indicates that either the drag pixel threshold or the
20028          * mousdown time threshold has been met
20029          * @property dragThreshMet
20030          * @type boolean
20031          * @private
20032          * @static
20033          */
20034         dragThreshMet: false,
20035
20036         /**
20037          * Timeout used for the click time threshold
20038          * @property clickTimeout
20039          * @type Object
20040          * @private
20041          * @static
20042          */
20043         clickTimeout: null,
20044
20045         /**
20046          * The X position of the mousedown event stored for later use when a
20047          * drag threshold is met.
20048          * @property startX
20049          * @type int
20050          * @private
20051          * @static
20052          */
20053         startX: 0,
20054
20055         /**
20056          * The Y position of the mousedown event stored for later use when a
20057          * drag threshold is met.
20058          * @property startY
20059          * @type int
20060          * @private
20061          * @static
20062          */
20063         startY: 0,
20064
20065         /**
20066          * Each DragDrop instance must be registered with the DragDropMgr.
20067          * This is executed in DragDrop.init()
20068          * @method regDragDrop
20069          * @param {DragDrop} oDD the DragDrop object to register
20070          * @param {String} sGroup the name of the group this element belongs to
20071          * @static
20072          */
20073         regDragDrop: function(oDD, sGroup) {
20074             if (!this.initialized) { this.init(); }
20075
20076             if (!this.ids[sGroup]) {
20077                 this.ids[sGroup] = {};
20078             }
20079             this.ids[sGroup][oDD.id] = oDD;
20080         },
20081
20082         /**
20083          * Removes the supplied dd instance from the supplied group. Executed
20084          * by DragDrop.removeFromGroup, so don't call this function directly.
20085          * @method removeDDFromGroup
20086          * @private
20087          * @static
20088          */
20089         removeDDFromGroup: function(oDD, sGroup) {
20090             if (!this.ids[sGroup]) {
20091                 this.ids[sGroup] = {};
20092             }
20093
20094             var obj = this.ids[sGroup];
20095             if (obj && obj[oDD.id]) {
20096                 delete obj[oDD.id];
20097             }
20098         },
20099
20100         /**
20101          * Unregisters a drag and drop item.  This is executed in
20102          * DragDrop.unreg, use that method instead of calling this directly.
20103          * @method _remove
20104          * @private
20105          * @static
20106          */
20107         _remove: function(oDD) {
20108             for (var g in oDD.groups) {
20109                 if (g && this.ids[g][oDD.id]) {
20110                     delete this.ids[g][oDD.id];
20111                 }
20112             }
20113             delete this.handleIds[oDD.id];
20114         },
20115
20116         /**
20117          * Each DragDrop handle element must be registered.  This is done
20118          * automatically when executing DragDrop.setHandleElId()
20119          * @method regHandle
20120          * @param {String} sDDId the DragDrop id this element is a handle for
20121          * @param {String} sHandleId the id of the element that is the drag
20122          * handle
20123          * @static
20124          */
20125         regHandle: function(sDDId, sHandleId) {
20126             if (!this.handleIds[sDDId]) {
20127                 this.handleIds[sDDId] = {};
20128             }
20129             this.handleIds[sDDId][sHandleId] = sHandleId;
20130         },
20131
20132         /**
20133          * Utility function to determine if a given element has been
20134          * registered as a drag drop item.
20135          * @method isDragDrop
20136          * @param {String} id the element id to check
20137          * @return {boolean} true if this element is a DragDrop item,
20138          * false otherwise
20139          * @static
20140          */
20141         isDragDrop: function(id) {
20142             return ( this.getDDById(id) ) ? true : false;
20143         },
20144
20145         /**
20146          * Returns the drag and drop instances that are in all groups the
20147          * passed in instance belongs to.
20148          * @method getRelated
20149          * @param {DragDrop} p_oDD the obj to get related data for
20150          * @param {boolean} bTargetsOnly if true, only return targetable objs
20151          * @return {DragDrop[]} the related instances
20152          * @static
20153          */
20154         getRelated: function(p_oDD, bTargetsOnly) {
20155             var oDDs = [];
20156             for (var i in p_oDD.groups) {
20157                 for (j in this.ids[i]) {
20158                     var dd = this.ids[i][j];
20159                     if (! this.isTypeOfDD(dd)) {
20160                         continue;
20161                     }
20162                     if (!bTargetsOnly || dd.isTarget) {
20163                         oDDs[oDDs.length] = dd;
20164                     }
20165                 }
20166             }
20167
20168             return oDDs;
20169         },
20170
20171         /**
20172          * Returns true if the specified dd target is a legal target for
20173          * the specifice drag obj
20174          * @method isLegalTarget
20175          * @param {DragDrop} the drag obj
20176          * @param {DragDrop} the target
20177          * @return {boolean} true if the target is a legal target for the
20178          * dd obj
20179          * @static
20180          */
20181         isLegalTarget: function (oDD, oTargetDD) {
20182             var targets = this.getRelated(oDD, true);
20183             for (var i=0, len=targets.length;i<len;++i) {
20184                 if (targets[i].id == oTargetDD.id) {
20185                     return true;
20186                 }
20187             }
20188
20189             return false;
20190         },
20191
20192         /**
20193          * My goal is to be able to transparently determine if an object is
20194          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
20195          * returns "object", oDD.constructor.toString() always returns
20196          * "DragDrop" and not the name of the subclass.  So for now it just
20197          * evaluates a well-known variable in DragDrop.
20198          * @method isTypeOfDD
20199          * @param {Object} the object to evaluate
20200          * @return {boolean} true if typeof oDD = DragDrop
20201          * @static
20202          */
20203         isTypeOfDD: function (oDD) {
20204             return (oDD && oDD.__ygDragDrop);
20205         },
20206
20207         /**
20208          * Utility function to determine if a given element has been
20209          * registered as a drag drop handle for the given Drag Drop object.
20210          * @method isHandle
20211          * @param {String} id the element id to check
20212          * @return {boolean} true if this element is a DragDrop handle, false
20213          * otherwise
20214          * @static
20215          */
20216         isHandle: function(sDDId, sHandleId) {
20217             return ( this.handleIds[sDDId] &&
20218                             this.handleIds[sDDId][sHandleId] );
20219         },
20220
20221         /**
20222          * Returns the DragDrop instance for a given id
20223          * @method getDDById
20224          * @param {String} id the id of the DragDrop object
20225          * @return {DragDrop} the drag drop object, null if it is not found
20226          * @static
20227          */
20228         getDDById: function(id) {
20229             for (var i in this.ids) {
20230                 if (this.ids[i][id]) {
20231                     return this.ids[i][id];
20232                 }
20233             }
20234             return null;
20235         },
20236
20237         /**
20238          * Fired after a registered DragDrop object gets the mousedown event.
20239          * Sets up the events required to track the object being dragged
20240          * @method handleMouseDown
20241          * @param {Event} e the event
20242          * @param oDD the DragDrop object being dragged
20243          * @private
20244          * @static
20245          */
20246         handleMouseDown: function(e, oDD) {
20247             if(Roo.QuickTips){
20248                 Roo.QuickTips.disable();
20249             }
20250             this.currentTarget = e.getTarget();
20251
20252             this.dragCurrent = oDD;
20253
20254             var el = oDD.getEl();
20255
20256             // track start position
20257             this.startX = e.getPageX();
20258             this.startY = e.getPageY();
20259
20260             this.deltaX = this.startX - el.offsetLeft;
20261             this.deltaY = this.startY - el.offsetTop;
20262
20263             this.dragThreshMet = false;
20264
20265             this.clickTimeout = setTimeout(
20266                     function() {
20267                         var DDM = Roo.dd.DDM;
20268                         DDM.startDrag(DDM.startX, DDM.startY);
20269                     },
20270                     this.clickTimeThresh );
20271         },
20272
20273         /**
20274          * Fired when either the drag pixel threshol or the mousedown hold
20275          * time threshold has been met.
20276          * @method startDrag
20277          * @param x {int} the X position of the original mousedown
20278          * @param y {int} the Y position of the original mousedown
20279          * @static
20280          */
20281         startDrag: function(x, y) {
20282             clearTimeout(this.clickTimeout);
20283             if (this.dragCurrent) {
20284                 this.dragCurrent.b4StartDrag(x, y);
20285                 this.dragCurrent.startDrag(x, y);
20286             }
20287             this.dragThreshMet = true;
20288         },
20289
20290         /**
20291          * Internal function to handle the mouseup event.  Will be invoked
20292          * from the context of the document.
20293          * @method handleMouseUp
20294          * @param {Event} e the event
20295          * @private
20296          * @static
20297          */
20298         handleMouseUp: function(e) {
20299
20300             if(Roo.QuickTips){
20301                 Roo.QuickTips.enable();
20302             }
20303             if (! this.dragCurrent) {
20304                 return;
20305             }
20306
20307             clearTimeout(this.clickTimeout);
20308
20309             if (this.dragThreshMet) {
20310                 this.fireEvents(e, true);
20311             } else {
20312             }
20313
20314             this.stopDrag(e);
20315
20316             this.stopEvent(e);
20317         },
20318
20319         /**
20320          * Utility to stop event propagation and event default, if these
20321          * features are turned on.
20322          * @method stopEvent
20323          * @param {Event} e the event as returned by this.getEvent()
20324          * @static
20325          */
20326         stopEvent: function(e){
20327             if(this.stopPropagation) {
20328                 e.stopPropagation();
20329             }
20330
20331             if (this.preventDefault) {
20332                 e.preventDefault();
20333             }
20334         },
20335
20336         /**
20337          * Internal function to clean up event handlers after the drag
20338          * operation is complete
20339          * @method stopDrag
20340          * @param {Event} e the event
20341          * @private
20342          * @static
20343          */
20344         stopDrag: function(e) {
20345             // Fire the drag end event for the item that was dragged
20346             if (this.dragCurrent) {
20347                 if (this.dragThreshMet) {
20348                     this.dragCurrent.b4EndDrag(e);
20349                     this.dragCurrent.endDrag(e);
20350                 }
20351
20352                 this.dragCurrent.onMouseUp(e);
20353             }
20354
20355             this.dragCurrent = null;
20356             this.dragOvers = {};
20357         },
20358
20359         /**
20360          * Internal function to handle the mousemove event.  Will be invoked
20361          * from the context of the html element.
20362          *
20363          * @TODO figure out what we can do about mouse events lost when the
20364          * user drags objects beyond the window boundary.  Currently we can
20365          * detect this in internet explorer by verifying that the mouse is
20366          * down during the mousemove event.  Firefox doesn't give us the
20367          * button state on the mousemove event.
20368          * @method handleMouseMove
20369          * @param {Event} e the event
20370          * @private
20371          * @static
20372          */
20373         handleMouseMove: function(e) {
20374             if (! this.dragCurrent) {
20375                 return true;
20376             }
20377
20378             // var button = e.which || e.button;
20379
20380             // check for IE mouseup outside of page boundary
20381             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
20382                 this.stopEvent(e);
20383                 return this.handleMouseUp(e);
20384             }
20385
20386             if (!this.dragThreshMet) {
20387                 var diffX = Math.abs(this.startX - e.getPageX());
20388                 var diffY = Math.abs(this.startY - e.getPageY());
20389                 if (diffX > this.clickPixelThresh ||
20390                             diffY > this.clickPixelThresh) {
20391                     this.startDrag(this.startX, this.startY);
20392                 }
20393             }
20394
20395             if (this.dragThreshMet) {
20396                 this.dragCurrent.b4Drag(e);
20397                 this.dragCurrent.onDrag(e);
20398                 if(!this.dragCurrent.moveOnly){
20399                     this.fireEvents(e, false);
20400                 }
20401             }
20402
20403             this.stopEvent(e);
20404
20405             return true;
20406         },
20407
20408         /**
20409          * Iterates over all of the DragDrop elements to find ones we are
20410          * hovering over or dropping on
20411          * @method fireEvents
20412          * @param {Event} e the event
20413          * @param {boolean} isDrop is this a drop op or a mouseover op?
20414          * @private
20415          * @static
20416          */
20417         fireEvents: function(e, isDrop) {
20418             var dc = this.dragCurrent;
20419
20420             // If the user did the mouse up outside of the window, we could
20421             // get here even though we have ended the drag.
20422             if (!dc || dc.isLocked()) {
20423                 return;
20424             }
20425
20426             var pt = e.getPoint();
20427
20428             // cache the previous dragOver array
20429             var oldOvers = [];
20430
20431             var outEvts   = [];
20432             var overEvts  = [];
20433             var dropEvts  = [];
20434             var enterEvts = [];
20435
20436             // Check to see if the object(s) we were hovering over is no longer
20437             // being hovered over so we can fire the onDragOut event
20438             for (var i in this.dragOvers) {
20439
20440                 var ddo = this.dragOvers[i];
20441
20442                 if (! this.isTypeOfDD(ddo)) {
20443                     continue;
20444                 }
20445
20446                 if (! this.isOverTarget(pt, ddo, this.mode)) {
20447                     outEvts.push( ddo );
20448                 }
20449
20450                 oldOvers[i] = true;
20451                 delete this.dragOvers[i];
20452             }
20453
20454             for (var sGroup in dc.groups) {
20455
20456                 if ("string" != typeof sGroup) {
20457                     continue;
20458                 }
20459
20460                 for (i in this.ids[sGroup]) {
20461                     var oDD = this.ids[sGroup][i];
20462                     if (! this.isTypeOfDD(oDD)) {
20463                         continue;
20464                     }
20465
20466                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
20467                         if (this.isOverTarget(pt, oDD, this.mode)) {
20468                             // look for drop interactions
20469                             if (isDrop) {
20470                                 dropEvts.push( oDD );
20471                             // look for drag enter and drag over interactions
20472                             } else {
20473
20474                                 // initial drag over: dragEnter fires
20475                                 if (!oldOvers[oDD.id]) {
20476                                     enterEvts.push( oDD );
20477                                 // subsequent drag overs: dragOver fires
20478                                 } else {
20479                                     overEvts.push( oDD );
20480                                 }
20481
20482                                 this.dragOvers[oDD.id] = oDD;
20483                             }
20484                         }
20485                     }
20486                 }
20487             }
20488
20489             if (this.mode) {
20490                 if (outEvts.length) {
20491                     dc.b4DragOut(e, outEvts);
20492                     dc.onDragOut(e, outEvts);
20493                 }
20494
20495                 if (enterEvts.length) {
20496                     dc.onDragEnter(e, enterEvts);
20497                 }
20498
20499                 if (overEvts.length) {
20500                     dc.b4DragOver(e, overEvts);
20501                     dc.onDragOver(e, overEvts);
20502                 }
20503
20504                 if (dropEvts.length) {
20505                     dc.b4DragDrop(e, dropEvts);
20506                     dc.onDragDrop(e, dropEvts);
20507                 }
20508
20509             } else {
20510                 // fire dragout events
20511                 var len = 0;
20512                 for (i=0, len=outEvts.length; i<len; ++i) {
20513                     dc.b4DragOut(e, outEvts[i].id);
20514                     dc.onDragOut(e, outEvts[i].id);
20515                 }
20516
20517                 // fire enter events
20518                 for (i=0,len=enterEvts.length; i<len; ++i) {
20519                     // dc.b4DragEnter(e, oDD.id);
20520                     dc.onDragEnter(e, enterEvts[i].id);
20521                 }
20522
20523                 // fire over events
20524                 for (i=0,len=overEvts.length; i<len; ++i) {
20525                     dc.b4DragOver(e, overEvts[i].id);
20526                     dc.onDragOver(e, overEvts[i].id);
20527                 }
20528
20529                 // fire drop events
20530                 for (i=0, len=dropEvts.length; i<len; ++i) {
20531                     dc.b4DragDrop(e, dropEvts[i].id);
20532                     dc.onDragDrop(e, dropEvts[i].id);
20533                 }
20534
20535             }
20536
20537             // notify about a drop that did not find a target
20538             if (isDrop && !dropEvts.length) {
20539                 dc.onInvalidDrop(e);
20540             }
20541
20542         },
20543
20544         /**
20545          * Helper function for getting the best match from the list of drag
20546          * and drop objects returned by the drag and drop events when we are
20547          * in INTERSECT mode.  It returns either the first object that the
20548          * cursor is over, or the object that has the greatest overlap with
20549          * the dragged element.
20550          * @method getBestMatch
20551          * @param  {DragDrop[]} dds The array of drag and drop objects
20552          * targeted
20553          * @return {DragDrop}       The best single match
20554          * @static
20555          */
20556         getBestMatch: function(dds) {
20557             var winner = null;
20558             // Return null if the input is not what we expect
20559             //if (!dds || !dds.length || dds.length == 0) {
20560                // winner = null;
20561             // If there is only one item, it wins
20562             //} else if (dds.length == 1) {
20563
20564             var len = dds.length;
20565
20566             if (len == 1) {
20567                 winner = dds[0];
20568             } else {
20569                 // Loop through the targeted items
20570                 for (var i=0; i<len; ++i) {
20571                     var dd = dds[i];
20572                     // If the cursor is over the object, it wins.  If the
20573                     // cursor is over multiple matches, the first one we come
20574                     // to wins.
20575                     if (dd.cursorIsOver) {
20576                         winner = dd;
20577                         break;
20578                     // Otherwise the object with the most overlap wins
20579                     } else {
20580                         if (!winner ||
20581                             winner.overlap.getArea() < dd.overlap.getArea()) {
20582                             winner = dd;
20583                         }
20584                     }
20585                 }
20586             }
20587
20588             return winner;
20589         },
20590
20591         /**
20592          * Refreshes the cache of the top-left and bottom-right points of the
20593          * drag and drop objects in the specified group(s).  This is in the
20594          * format that is stored in the drag and drop instance, so typical
20595          * usage is:
20596          * <code>
20597          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20598          * </code>
20599          * Alternatively:
20600          * <code>
20601          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20602          * </code>
20603          * @TODO this really should be an indexed array.  Alternatively this
20604          * method could accept both.
20605          * @method refreshCache
20606          * @param {Object} groups an associative array of groups to refresh
20607          * @static
20608          */
20609         refreshCache: function(groups) {
20610             for (var sGroup in groups) {
20611                 if ("string" != typeof sGroup) {
20612                     continue;
20613                 }
20614                 for (var i in this.ids[sGroup]) {
20615                     var oDD = this.ids[sGroup][i];
20616
20617                     if (this.isTypeOfDD(oDD)) {
20618                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20619                         var loc = this.getLocation(oDD);
20620                         if (loc) {
20621                             this.locationCache[oDD.id] = loc;
20622                         } else {
20623                             delete this.locationCache[oDD.id];
20624                             // this will unregister the drag and drop object if
20625                             // the element is not in a usable state
20626                             // oDD.unreg();
20627                         }
20628                     }
20629                 }
20630             }
20631         },
20632
20633         /**
20634          * This checks to make sure an element exists and is in the DOM.  The
20635          * main purpose is to handle cases where innerHTML is used to remove
20636          * drag and drop objects from the DOM.  IE provides an 'unspecified
20637          * error' when trying to access the offsetParent of such an element
20638          * @method verifyEl
20639          * @param {HTMLElement} el the element to check
20640          * @return {boolean} true if the element looks usable
20641          * @static
20642          */
20643         verifyEl: function(el) {
20644             if (el) {
20645                 var parent;
20646                 if(Roo.isIE){
20647                     try{
20648                         parent = el.offsetParent;
20649                     }catch(e){}
20650                 }else{
20651                     parent = el.offsetParent;
20652                 }
20653                 if (parent) {
20654                     return true;
20655                 }
20656             }
20657
20658             return false;
20659         },
20660
20661         /**
20662          * Returns a Region object containing the drag and drop element's position
20663          * and size, including the padding configured for it
20664          * @method getLocation
20665          * @param {DragDrop} oDD the drag and drop object to get the
20666          *                       location for
20667          * @return {Roo.lib.Region} a Region object representing the total area
20668          *                             the element occupies, including any padding
20669          *                             the instance is configured for.
20670          * @static
20671          */
20672         getLocation: function(oDD) {
20673             if (! this.isTypeOfDD(oDD)) {
20674                 return null;
20675             }
20676
20677             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20678
20679             try {
20680                 pos= Roo.lib.Dom.getXY(el);
20681             } catch (e) { }
20682
20683             if (!pos) {
20684                 return null;
20685             }
20686
20687             x1 = pos[0];
20688             x2 = x1 + el.offsetWidth;
20689             y1 = pos[1];
20690             y2 = y1 + el.offsetHeight;
20691
20692             t = y1 - oDD.padding[0];
20693             r = x2 + oDD.padding[1];
20694             b = y2 + oDD.padding[2];
20695             l = x1 - oDD.padding[3];
20696
20697             return new Roo.lib.Region( t, r, b, l );
20698         },
20699
20700         /**
20701          * Checks the cursor location to see if it over the target
20702          * @method isOverTarget
20703          * @param {Roo.lib.Point} pt The point to evaluate
20704          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20705          * @return {boolean} true if the mouse is over the target
20706          * @private
20707          * @static
20708          */
20709         isOverTarget: function(pt, oTarget, intersect) {
20710             // use cache if available
20711             var loc = this.locationCache[oTarget.id];
20712             if (!loc || !this.useCache) {
20713                 loc = this.getLocation(oTarget);
20714                 this.locationCache[oTarget.id] = loc;
20715
20716             }
20717
20718             if (!loc) {
20719                 return false;
20720             }
20721
20722             oTarget.cursorIsOver = loc.contains( pt );
20723
20724             // DragDrop is using this as a sanity check for the initial mousedown
20725             // in this case we are done.  In POINT mode, if the drag obj has no
20726             // contraints, we are also done. Otherwise we need to evaluate the
20727             // location of the target as related to the actual location of the
20728             // dragged element.
20729             var dc = this.dragCurrent;
20730             if (!dc || !dc.getTargetCoord ||
20731                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20732                 return oTarget.cursorIsOver;
20733             }
20734
20735             oTarget.overlap = null;
20736
20737             // Get the current location of the drag element, this is the
20738             // location of the mouse event less the delta that represents
20739             // where the original mousedown happened on the element.  We
20740             // need to consider constraints and ticks as well.
20741             var pos = dc.getTargetCoord(pt.x, pt.y);
20742
20743             var el = dc.getDragEl();
20744             var curRegion = new Roo.lib.Region( pos.y,
20745                                                    pos.x + el.offsetWidth,
20746                                                    pos.y + el.offsetHeight,
20747                                                    pos.x );
20748
20749             var overlap = curRegion.intersect(loc);
20750
20751             if (overlap) {
20752                 oTarget.overlap = overlap;
20753                 return (intersect) ? true : oTarget.cursorIsOver;
20754             } else {
20755                 return false;
20756             }
20757         },
20758
20759         /**
20760          * unload event handler
20761          * @method _onUnload
20762          * @private
20763          * @static
20764          */
20765         _onUnload: function(e, me) {
20766             Roo.dd.DragDropMgr.unregAll();
20767         },
20768
20769         /**
20770          * Cleans up the drag and drop events and objects.
20771          * @method unregAll
20772          * @private
20773          * @static
20774          */
20775         unregAll: function() {
20776
20777             if (this.dragCurrent) {
20778                 this.stopDrag();
20779                 this.dragCurrent = null;
20780             }
20781
20782             this._execOnAll("unreg", []);
20783
20784             for (i in this.elementCache) {
20785                 delete this.elementCache[i];
20786             }
20787
20788             this.elementCache = {};
20789             this.ids = {};
20790         },
20791
20792         /**
20793          * A cache of DOM elements
20794          * @property elementCache
20795          * @private
20796          * @static
20797          */
20798         elementCache: {},
20799
20800         /**
20801          * Get the wrapper for the DOM element specified
20802          * @method getElWrapper
20803          * @param {String} id the id of the element to get
20804          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20805          * @private
20806          * @deprecated This wrapper isn't that useful
20807          * @static
20808          */
20809         getElWrapper: function(id) {
20810             var oWrapper = this.elementCache[id];
20811             if (!oWrapper || !oWrapper.el) {
20812                 oWrapper = this.elementCache[id] =
20813                     new this.ElementWrapper(Roo.getDom(id));
20814             }
20815             return oWrapper;
20816         },
20817
20818         /**
20819          * Returns the actual DOM element
20820          * @method getElement
20821          * @param {String} id the id of the elment to get
20822          * @return {Object} The element
20823          * @deprecated use Roo.getDom instead
20824          * @static
20825          */
20826         getElement: function(id) {
20827             return Roo.getDom(id);
20828         },
20829
20830         /**
20831          * Returns the style property for the DOM element (i.e.,
20832          * document.getElById(id).style)
20833          * @method getCss
20834          * @param {String} id the id of the elment to get
20835          * @return {Object} The style property of the element
20836          * @deprecated use Roo.getDom instead
20837          * @static
20838          */
20839         getCss: function(id) {
20840             var el = Roo.getDom(id);
20841             return (el) ? el.style : null;
20842         },
20843
20844         /**
20845          * Inner class for cached elements
20846          * @class DragDropMgr.ElementWrapper
20847          * @for DragDropMgr
20848          * @private
20849          * @deprecated
20850          */
20851         ElementWrapper: function(el) {
20852                 /**
20853                  * The element
20854                  * @property el
20855                  */
20856                 this.el = el || null;
20857                 /**
20858                  * The element id
20859                  * @property id
20860                  */
20861                 this.id = this.el && el.id;
20862                 /**
20863                  * A reference to the style property
20864                  * @property css
20865                  */
20866                 this.css = this.el && el.style;
20867             },
20868
20869         /**
20870          * Returns the X position of an html element
20871          * @method getPosX
20872          * @param el the element for which to get the position
20873          * @return {int} the X coordinate
20874          * @for DragDropMgr
20875          * @deprecated use Roo.lib.Dom.getX instead
20876          * @static
20877          */
20878         getPosX: function(el) {
20879             return Roo.lib.Dom.getX(el);
20880         },
20881
20882         /**
20883          * Returns the Y position of an html element
20884          * @method getPosY
20885          * @param el the element for which to get the position
20886          * @return {int} the Y coordinate
20887          * @deprecated use Roo.lib.Dom.getY instead
20888          * @static
20889          */
20890         getPosY: function(el) {
20891             return Roo.lib.Dom.getY(el);
20892         },
20893
20894         /**
20895          * Swap two nodes.  In IE, we use the native method, for others we
20896          * emulate the IE behavior
20897          * @method swapNode
20898          * @param n1 the first node to swap
20899          * @param n2 the other node to swap
20900          * @static
20901          */
20902         swapNode: function(n1, n2) {
20903             if (n1.swapNode) {
20904                 n1.swapNode(n2);
20905             } else {
20906                 var p = n2.parentNode;
20907                 var s = n2.nextSibling;
20908
20909                 if (s == n1) {
20910                     p.insertBefore(n1, n2);
20911                 } else if (n2 == n1.nextSibling) {
20912                     p.insertBefore(n2, n1);
20913                 } else {
20914                     n1.parentNode.replaceChild(n2, n1);
20915                     p.insertBefore(n1, s);
20916                 }
20917             }
20918         },
20919
20920         /**
20921          * Returns the current scroll position
20922          * @method getScroll
20923          * @private
20924          * @static
20925          */
20926         getScroll: function () {
20927             var t, l, dde=document.documentElement, db=document.body;
20928             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20929                 t = dde.scrollTop;
20930                 l = dde.scrollLeft;
20931             } else if (db) {
20932                 t = db.scrollTop;
20933                 l = db.scrollLeft;
20934             } else {
20935
20936             }
20937             return { top: t, left: l };
20938         },
20939
20940         /**
20941          * Returns the specified element style property
20942          * @method getStyle
20943          * @param {HTMLElement} el          the element
20944          * @param {string}      styleProp   the style property
20945          * @return {string} The value of the style property
20946          * @deprecated use Roo.lib.Dom.getStyle
20947          * @static
20948          */
20949         getStyle: function(el, styleProp) {
20950             return Roo.fly(el).getStyle(styleProp);
20951         },
20952
20953         /**
20954          * Gets the scrollTop
20955          * @method getScrollTop
20956          * @return {int} the document's scrollTop
20957          * @static
20958          */
20959         getScrollTop: function () { return this.getScroll().top; },
20960
20961         /**
20962          * Gets the scrollLeft
20963          * @method getScrollLeft
20964          * @return {int} the document's scrollTop
20965          * @static
20966          */
20967         getScrollLeft: function () { return this.getScroll().left; },
20968
20969         /**
20970          * Sets the x/y position of an element to the location of the
20971          * target element.
20972          * @method moveToEl
20973          * @param {HTMLElement} moveEl      The element to move
20974          * @param {HTMLElement} targetEl    The position reference element
20975          * @static
20976          */
20977         moveToEl: function (moveEl, targetEl) {
20978             var aCoord = Roo.lib.Dom.getXY(targetEl);
20979             Roo.lib.Dom.setXY(moveEl, aCoord);
20980         },
20981
20982         /**
20983          * Numeric array sort function
20984          * @method numericSort
20985          * @static
20986          */
20987         numericSort: function(a, b) { return (a - b); },
20988
20989         /**
20990          * Internal counter
20991          * @property _timeoutCount
20992          * @private
20993          * @static
20994          */
20995         _timeoutCount: 0,
20996
20997         /**
20998          * Trying to make the load order less important.  Without this we get
20999          * an error if this file is loaded before the Event Utility.
21000          * @method _addListeners
21001          * @private
21002          * @static
21003          */
21004         _addListeners: function() {
21005             var DDM = Roo.dd.DDM;
21006             if ( Roo.lib.Event && document ) {
21007                 DDM._onLoad();
21008             } else {
21009                 if (DDM._timeoutCount > 2000) {
21010                 } else {
21011                     setTimeout(DDM._addListeners, 10);
21012                     if (document && document.body) {
21013                         DDM._timeoutCount += 1;
21014                     }
21015                 }
21016             }
21017         },
21018
21019         /**
21020          * Recursively searches the immediate parent and all child nodes for
21021          * the handle element in order to determine wheter or not it was
21022          * clicked.
21023          * @method handleWasClicked
21024          * @param node the html element to inspect
21025          * @static
21026          */
21027         handleWasClicked: function(node, id) {
21028             if (this.isHandle(id, node.id)) {
21029                 return true;
21030             } else {
21031                 // check to see if this is a text node child of the one we want
21032                 var p = node.parentNode;
21033
21034                 while (p) {
21035                     if (this.isHandle(id, p.id)) {
21036                         return true;
21037                     } else {
21038                         p = p.parentNode;
21039                     }
21040                 }
21041             }
21042
21043             return false;
21044         }
21045
21046     };
21047
21048 }();
21049
21050 // shorter alias, save a few bytes
21051 Roo.dd.DDM = Roo.dd.DragDropMgr;
21052 Roo.dd.DDM._addListeners();
21053
21054 }/*
21055  * Based on:
21056  * Ext JS Library 1.1.1
21057  * Copyright(c) 2006-2007, Ext JS, LLC.
21058  *
21059  * Originally Released Under LGPL - original licence link has changed is not relivant.
21060  *
21061  * Fork - LGPL
21062  * <script type="text/javascript">
21063  */
21064
21065 /**
21066  * @class Roo.dd.DD
21067  * A DragDrop implementation where the linked element follows the
21068  * mouse cursor during a drag.
21069  * @extends Roo.dd.DragDrop
21070  * @constructor
21071  * @param {String} id the id of the linked element
21072  * @param {String} sGroup the group of related DragDrop items
21073  * @param {object} config an object containing configurable attributes
21074  *                Valid properties for DD:
21075  *                    scroll
21076  */
21077 Roo.dd.DD = function(id, sGroup, config) {
21078     if (id) {
21079         this.init(id, sGroup, config);
21080     }
21081 };
21082
21083 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
21084
21085     /**
21086      * When set to true, the utility automatically tries to scroll the browser
21087      * window wehn a drag and drop element is dragged near the viewport boundary.
21088      * Defaults to true.
21089      * @property scroll
21090      * @type boolean
21091      */
21092     scroll: true,
21093
21094     /**
21095      * Sets the pointer offset to the distance between the linked element's top
21096      * left corner and the location the element was clicked
21097      * @method autoOffset
21098      * @param {int} iPageX the X coordinate of the click
21099      * @param {int} iPageY the Y coordinate of the click
21100      */
21101     autoOffset: function(iPageX, iPageY) {
21102         var x = iPageX - this.startPageX;
21103         var y = iPageY - this.startPageY;
21104         this.setDelta(x, y);
21105     },
21106
21107     /**
21108      * Sets the pointer offset.  You can call this directly to force the
21109      * offset to be in a particular location (e.g., pass in 0,0 to set it
21110      * to the center of the object)
21111      * @method setDelta
21112      * @param {int} iDeltaX the distance from the left
21113      * @param {int} iDeltaY the distance from the top
21114      */
21115     setDelta: function(iDeltaX, iDeltaY) {
21116         this.deltaX = iDeltaX;
21117         this.deltaY = iDeltaY;
21118     },
21119
21120     /**
21121      * Sets the drag element to the location of the mousedown or click event,
21122      * maintaining the cursor location relative to the location on the element
21123      * that was clicked.  Override this if you want to place the element in a
21124      * location other than where the cursor is.
21125      * @method setDragElPos
21126      * @param {int} iPageX the X coordinate of the mousedown or drag event
21127      * @param {int} iPageY the Y coordinate of the mousedown or drag event
21128      */
21129     setDragElPos: function(iPageX, iPageY) {
21130         // the first time we do this, we are going to check to make sure
21131         // the element has css positioning
21132
21133         var el = this.getDragEl();
21134         this.alignElWithMouse(el, iPageX, iPageY);
21135     },
21136
21137     /**
21138      * Sets the element to the location of the mousedown or click event,
21139      * maintaining the cursor location relative to the location on the element
21140      * that was clicked.  Override this if you want to place the element in a
21141      * location other than where the cursor is.
21142      * @method alignElWithMouse
21143      * @param {HTMLElement} el the element to move
21144      * @param {int} iPageX the X coordinate of the mousedown or drag event
21145      * @param {int} iPageY the Y coordinate of the mousedown or drag event
21146      */
21147     alignElWithMouse: function(el, iPageX, iPageY) {
21148         var oCoord = this.getTargetCoord(iPageX, iPageY);
21149         var fly = el.dom ? el : Roo.fly(el);
21150         if (!this.deltaSetXY) {
21151             var aCoord = [oCoord.x, oCoord.y];
21152             fly.setXY(aCoord);
21153             var newLeft = fly.getLeft(true);
21154             var newTop  = fly.getTop(true);
21155             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
21156         } else {
21157             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
21158         }
21159
21160         this.cachePosition(oCoord.x, oCoord.y);
21161         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
21162         return oCoord;
21163     },
21164
21165     /**
21166      * Saves the most recent position so that we can reset the constraints and
21167      * tick marks on-demand.  We need to know this so that we can calculate the
21168      * number of pixels the element is offset from its original position.
21169      * @method cachePosition
21170      * @param iPageX the current x position (optional, this just makes it so we
21171      * don't have to look it up again)
21172      * @param iPageY the current y position (optional, this just makes it so we
21173      * don't have to look it up again)
21174      */
21175     cachePosition: function(iPageX, iPageY) {
21176         if (iPageX) {
21177             this.lastPageX = iPageX;
21178             this.lastPageY = iPageY;
21179         } else {
21180             var aCoord = Roo.lib.Dom.getXY(this.getEl());
21181             this.lastPageX = aCoord[0];
21182             this.lastPageY = aCoord[1];
21183         }
21184     },
21185
21186     /**
21187      * Auto-scroll the window if the dragged object has been moved beyond the
21188      * visible window boundary.
21189      * @method autoScroll
21190      * @param {int} x the drag element's x position
21191      * @param {int} y the drag element's y position
21192      * @param {int} h the height of the drag element
21193      * @param {int} w the width of the drag element
21194      * @private
21195      */
21196     autoScroll: function(x, y, h, w) {
21197
21198         if (this.scroll) {
21199             // The client height
21200             var clientH = Roo.lib.Dom.getViewWidth();
21201
21202             // The client width
21203             var clientW = Roo.lib.Dom.getViewHeight();
21204
21205             // The amt scrolled down
21206             var st = this.DDM.getScrollTop();
21207
21208             // The amt scrolled right
21209             var sl = this.DDM.getScrollLeft();
21210
21211             // Location of the bottom of the element
21212             var bot = h + y;
21213
21214             // Location of the right of the element
21215             var right = w + x;
21216
21217             // The distance from the cursor to the bottom of the visible area,
21218             // adjusted so that we don't scroll if the cursor is beyond the
21219             // element drag constraints
21220             var toBot = (clientH + st - y - this.deltaY);
21221
21222             // The distance from the cursor to the right of the visible area
21223             var toRight = (clientW + sl - x - this.deltaX);
21224
21225
21226             // How close to the edge the cursor must be before we scroll
21227             // var thresh = (document.all) ? 100 : 40;
21228             var thresh = 40;
21229
21230             // How many pixels to scroll per autoscroll op.  This helps to reduce
21231             // clunky scrolling. IE is more sensitive about this ... it needs this
21232             // value to be higher.
21233             var scrAmt = (document.all) ? 80 : 30;
21234
21235             // Scroll down if we are near the bottom of the visible page and the
21236             // obj extends below the crease
21237             if ( bot > clientH && toBot < thresh ) {
21238                 window.scrollTo(sl, st + scrAmt);
21239             }
21240
21241             // Scroll up if the window is scrolled down and the top of the object
21242             // goes above the top border
21243             if ( y < st && st > 0 && y - st < thresh ) {
21244                 window.scrollTo(sl, st - scrAmt);
21245             }
21246
21247             // Scroll right if the obj is beyond the right border and the cursor is
21248             // near the border.
21249             if ( right > clientW && toRight < thresh ) {
21250                 window.scrollTo(sl + scrAmt, st);
21251             }
21252
21253             // Scroll left if the window has been scrolled to the right and the obj
21254             // extends past the left border
21255             if ( x < sl && sl > 0 && x - sl < thresh ) {
21256                 window.scrollTo(sl - scrAmt, st);
21257             }
21258         }
21259     },
21260
21261     /**
21262      * Finds the location the element should be placed if we want to move
21263      * it to where the mouse location less the click offset would place us.
21264      * @method getTargetCoord
21265      * @param {int} iPageX the X coordinate of the click
21266      * @param {int} iPageY the Y coordinate of the click
21267      * @return an object that contains the coordinates (Object.x and Object.y)
21268      * @private
21269      */
21270     getTargetCoord: function(iPageX, iPageY) {
21271
21272
21273         var x = iPageX - this.deltaX;
21274         var y = iPageY - this.deltaY;
21275
21276         if (this.constrainX) {
21277             if (x < this.minX) { x = this.minX; }
21278             if (x > this.maxX) { x = this.maxX; }
21279         }
21280
21281         if (this.constrainY) {
21282             if (y < this.minY) { y = this.minY; }
21283             if (y > this.maxY) { y = this.maxY; }
21284         }
21285
21286         x = this.getTick(x, this.xTicks);
21287         y = this.getTick(y, this.yTicks);
21288
21289
21290         return {x:x, y:y};
21291     },
21292
21293     /*
21294      * Sets up config options specific to this class. Overrides
21295      * Roo.dd.DragDrop, but all versions of this method through the
21296      * inheritance chain are called
21297      */
21298     applyConfig: function() {
21299         Roo.dd.DD.superclass.applyConfig.call(this);
21300         this.scroll = (this.config.scroll !== false);
21301     },
21302
21303     /*
21304      * Event that fires prior to the onMouseDown event.  Overrides
21305      * Roo.dd.DragDrop.
21306      */
21307     b4MouseDown: function(e) {
21308         // this.resetConstraints();
21309         this.autoOffset(e.getPageX(),
21310                             e.getPageY());
21311     },
21312
21313     /*
21314      * Event that fires prior to the onDrag event.  Overrides
21315      * Roo.dd.DragDrop.
21316      */
21317     b4Drag: function(e) {
21318         this.setDragElPos(e.getPageX(),
21319                             e.getPageY());
21320     },
21321
21322     toString: function() {
21323         return ("DD " + this.id);
21324     }
21325
21326     //////////////////////////////////////////////////////////////////////////
21327     // Debugging ygDragDrop events that can be overridden
21328     //////////////////////////////////////////////////////////////////////////
21329     /*
21330     startDrag: function(x, y) {
21331     },
21332
21333     onDrag: function(e) {
21334     },
21335
21336     onDragEnter: function(e, id) {
21337     },
21338
21339     onDragOver: function(e, id) {
21340     },
21341
21342     onDragOut: function(e, id) {
21343     },
21344
21345     onDragDrop: function(e, id) {
21346     },
21347
21348     endDrag: function(e) {
21349     }
21350
21351     */
21352
21353 });/*
21354  * Based on:
21355  * Ext JS Library 1.1.1
21356  * Copyright(c) 2006-2007, Ext JS, LLC.
21357  *
21358  * Originally Released Under LGPL - original licence link has changed is not relivant.
21359  *
21360  * Fork - LGPL
21361  * <script type="text/javascript">
21362  */
21363
21364 /**
21365  * @class Roo.dd.DDProxy
21366  * A DragDrop implementation that inserts an empty, bordered div into
21367  * the document that follows the cursor during drag operations.  At the time of
21368  * the click, the frame div is resized to the dimensions of the linked html
21369  * element, and moved to the exact location of the linked element.
21370  *
21371  * References to the "frame" element refer to the single proxy element that
21372  * was created to be dragged in place of all DDProxy elements on the
21373  * page.
21374  *
21375  * @extends Roo.dd.DD
21376  * @constructor
21377  * @param {String} id the id of the linked html element
21378  * @param {String} sGroup the group of related DragDrop objects
21379  * @param {object} config an object containing configurable attributes
21380  *                Valid properties for DDProxy in addition to those in DragDrop:
21381  *                   resizeFrame, centerFrame, dragElId
21382  */
21383 Roo.dd.DDProxy = function(id, sGroup, config) {
21384     if (id) {
21385         this.init(id, sGroup, config);
21386         this.initFrame();
21387     }
21388 };
21389
21390 /**
21391  * The default drag frame div id
21392  * @property Roo.dd.DDProxy.dragElId
21393  * @type String
21394  * @static
21395  */
21396 Roo.dd.DDProxy.dragElId = "ygddfdiv";
21397
21398 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
21399
21400     /**
21401      * By default we resize the drag frame to be the same size as the element
21402      * we want to drag (this is to get the frame effect).  We can turn it off
21403      * if we want a different behavior.
21404      * @property resizeFrame
21405      * @type boolean
21406      */
21407     resizeFrame: true,
21408
21409     /**
21410      * By default the frame is positioned exactly where the drag element is, so
21411      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
21412      * you do not have constraints on the obj is to have the drag frame centered
21413      * around the cursor.  Set centerFrame to true for this effect.
21414      * @property centerFrame
21415      * @type boolean
21416      */
21417     centerFrame: false,
21418
21419     /**
21420      * Creates the proxy element if it does not yet exist
21421      * @method createFrame
21422      */
21423     createFrame: function() {
21424         var self = this;
21425         var body = document.body;
21426
21427         if (!body || !body.firstChild) {
21428             setTimeout( function() { self.createFrame(); }, 50 );
21429             return;
21430         }
21431
21432         var div = this.getDragEl();
21433
21434         if (!div) {
21435             div    = document.createElement("div");
21436             div.id = this.dragElId;
21437             var s  = div.style;
21438
21439             s.position   = "absolute";
21440             s.visibility = "hidden";
21441             s.cursor     = "move";
21442             s.border     = "2px solid #aaa";
21443             s.zIndex     = 999;
21444
21445             // appendChild can blow up IE if invoked prior to the window load event
21446             // while rendering a table.  It is possible there are other scenarios
21447             // that would cause this to happen as well.
21448             body.insertBefore(div, body.firstChild);
21449         }
21450     },
21451
21452     /**
21453      * Initialization for the drag frame element.  Must be called in the
21454      * constructor of all subclasses
21455      * @method initFrame
21456      */
21457     initFrame: function() {
21458         this.createFrame();
21459     },
21460
21461     applyConfig: function() {
21462         Roo.dd.DDProxy.superclass.applyConfig.call(this);
21463
21464         this.resizeFrame = (this.config.resizeFrame !== false);
21465         this.centerFrame = (this.config.centerFrame);
21466         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
21467     },
21468
21469     /**
21470      * Resizes the drag frame to the dimensions of the clicked object, positions
21471      * it over the object, and finally displays it
21472      * @method showFrame
21473      * @param {int} iPageX X click position
21474      * @param {int} iPageY Y click position
21475      * @private
21476      */
21477     showFrame: function(iPageX, iPageY) {
21478         var el = this.getEl();
21479         var dragEl = this.getDragEl();
21480         var s = dragEl.style;
21481
21482         this._resizeProxy();
21483
21484         if (this.centerFrame) {
21485             this.setDelta( Math.round(parseInt(s.width,  10)/2),
21486                            Math.round(parseInt(s.height, 10)/2) );
21487         }
21488
21489         this.setDragElPos(iPageX, iPageY);
21490
21491         Roo.fly(dragEl).show();
21492     },
21493
21494     /**
21495      * The proxy is automatically resized to the dimensions of the linked
21496      * element when a drag is initiated, unless resizeFrame is set to false
21497      * @method _resizeProxy
21498      * @private
21499      */
21500     _resizeProxy: function() {
21501         if (this.resizeFrame) {
21502             var el = this.getEl();
21503             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21504         }
21505     },
21506
21507     // overrides Roo.dd.DragDrop
21508     b4MouseDown: function(e) {
21509         var x = e.getPageX();
21510         var y = e.getPageY();
21511         this.autoOffset(x, y);
21512         this.setDragElPos(x, y);
21513     },
21514
21515     // overrides Roo.dd.DragDrop
21516     b4StartDrag: function(x, y) {
21517         // show the drag frame
21518         this.showFrame(x, y);
21519     },
21520
21521     // overrides Roo.dd.DragDrop
21522     b4EndDrag: function(e) {
21523         Roo.fly(this.getDragEl()).hide();
21524     },
21525
21526     // overrides Roo.dd.DragDrop
21527     // By default we try to move the element to the last location of the frame.
21528     // This is so that the default behavior mirrors that of Roo.dd.DD.
21529     endDrag: function(e) {
21530
21531         var lel = this.getEl();
21532         var del = this.getDragEl();
21533
21534         // Show the drag frame briefly so we can get its position
21535         del.style.visibility = "";
21536
21537         this.beforeMove();
21538         // Hide the linked element before the move to get around a Safari
21539         // rendering bug.
21540         lel.style.visibility = "hidden";
21541         Roo.dd.DDM.moveToEl(lel, del);
21542         del.style.visibility = "hidden";
21543         lel.style.visibility = "";
21544
21545         this.afterDrag();
21546     },
21547
21548     beforeMove : function(){
21549
21550     },
21551
21552     afterDrag : function(){
21553
21554     },
21555
21556     toString: function() {
21557         return ("DDProxy " + this.id);
21558     }
21559
21560 });
21561 /*
21562  * Based on:
21563  * Ext JS Library 1.1.1
21564  * Copyright(c) 2006-2007, Ext JS, LLC.
21565  *
21566  * Originally Released Under LGPL - original licence link has changed is not relivant.
21567  *
21568  * Fork - LGPL
21569  * <script type="text/javascript">
21570  */
21571
21572  /**
21573  * @class Roo.dd.DDTarget
21574  * A DragDrop implementation that does not move, but can be a drop
21575  * target.  You would get the same result by simply omitting implementation
21576  * for the event callbacks, but this way we reduce the processing cost of the
21577  * event listener and the callbacks.
21578  * @extends Roo.dd.DragDrop
21579  * @constructor
21580  * @param {String} id the id of the element that is a drop target
21581  * @param {String} sGroup the group of related DragDrop objects
21582  * @param {object} config an object containing configurable attributes
21583  *                 Valid properties for DDTarget in addition to those in
21584  *                 DragDrop:
21585  *                    none
21586  */
21587 Roo.dd.DDTarget = function(id, sGroup, config) {
21588     if (id) {
21589         this.initTarget(id, sGroup, config);
21590     }
21591     if (config && (config.listeners || config.events)) { 
21592         Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21593             listeners : config.listeners || {}, 
21594             events : config.events || {} 
21595         });    
21596     }
21597 };
21598
21599 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21600 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21601     toString: function() {
21602         return ("DDTarget " + this.id);
21603     }
21604 });
21605 /*
21606  * Based on:
21607  * Ext JS Library 1.1.1
21608  * Copyright(c) 2006-2007, Ext JS, LLC.
21609  *
21610  * Originally Released Under LGPL - original licence link has changed is not relivant.
21611  *
21612  * Fork - LGPL
21613  * <script type="text/javascript">
21614  */
21615  
21616
21617 /**
21618  * @class Roo.dd.ScrollManager
21619  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21620  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21621  * @singleton
21622  */
21623 Roo.dd.ScrollManager = function(){
21624     var ddm = Roo.dd.DragDropMgr;
21625     var els = {};
21626     var dragEl = null;
21627     var proc = {};
21628     
21629     
21630     
21631     var onStop = function(e){
21632         dragEl = null;
21633         clearProc();
21634     };
21635     
21636     var triggerRefresh = function(){
21637         if(ddm.dragCurrent){
21638              ddm.refreshCache(ddm.dragCurrent.groups);
21639         }
21640     };
21641     
21642     var doScroll = function(){
21643         if(ddm.dragCurrent){
21644             var dds = Roo.dd.ScrollManager;
21645             if(!dds.animate){
21646                 if(proc.el.scroll(proc.dir, dds.increment)){
21647                     triggerRefresh();
21648                 }
21649             }else{
21650                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21651             }
21652         }
21653     };
21654     
21655     var clearProc = function(){
21656         if(proc.id){
21657             clearInterval(proc.id);
21658         }
21659         proc.id = 0;
21660         proc.el = null;
21661         proc.dir = "";
21662     };
21663     
21664     var startProc = function(el, dir){
21665          Roo.log('scroll startproc');
21666         clearProc();
21667         proc.el = el;
21668         proc.dir = dir;
21669         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21670     };
21671     
21672     var onFire = function(e, isDrop){
21673        
21674         if(isDrop || !ddm.dragCurrent){ return; }
21675         var dds = Roo.dd.ScrollManager;
21676         if(!dragEl || dragEl != ddm.dragCurrent){
21677             dragEl = ddm.dragCurrent;
21678             // refresh regions on drag start
21679             dds.refreshCache();
21680         }
21681         
21682         var xy = Roo.lib.Event.getXY(e);
21683         var pt = new Roo.lib.Point(xy[0], xy[1]);
21684         for(var id in els){
21685             var el = els[id], r = el._region;
21686             if(r && r.contains(pt) && el.isScrollable()){
21687                 if(r.bottom - pt.y <= dds.thresh){
21688                     if(proc.el != el){
21689                         startProc(el, "down");
21690                     }
21691                     return;
21692                 }else if(r.right - pt.x <= dds.thresh){
21693                     if(proc.el != el){
21694                         startProc(el, "left");
21695                     }
21696                     return;
21697                 }else if(pt.y - r.top <= dds.thresh){
21698                     if(proc.el != el){
21699                         startProc(el, "up");
21700                     }
21701                     return;
21702                 }else if(pt.x - r.left <= dds.thresh){
21703                     if(proc.el != el){
21704                         startProc(el, "right");
21705                     }
21706                     return;
21707                 }
21708             }
21709         }
21710         clearProc();
21711     };
21712     
21713     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21714     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21715     
21716     return {
21717         /**
21718          * Registers new overflow element(s) to auto scroll
21719          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21720          */
21721         register : function(el){
21722             if(el instanceof Array){
21723                 for(var i = 0, len = el.length; i < len; i++) {
21724                         this.register(el[i]);
21725                 }
21726             }else{
21727                 el = Roo.get(el);
21728                 els[el.id] = el;
21729             }
21730             Roo.dd.ScrollManager.els = els;
21731         },
21732         
21733         /**
21734          * Unregisters overflow element(s) so they are no longer scrolled
21735          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21736          */
21737         unregister : function(el){
21738             if(el instanceof Array){
21739                 for(var i = 0, len = el.length; i < len; i++) {
21740                         this.unregister(el[i]);
21741                 }
21742             }else{
21743                 el = Roo.get(el);
21744                 delete els[el.id];
21745             }
21746         },
21747         
21748         /**
21749          * The number of pixels from the edge of a container the pointer needs to be to 
21750          * trigger scrolling (defaults to 25)
21751          * @type Number
21752          */
21753         thresh : 25,
21754         
21755         /**
21756          * The number of pixels to scroll in each scroll increment (defaults to 50)
21757          * @type Number
21758          */
21759         increment : 100,
21760         
21761         /**
21762          * The frequency of scrolls in milliseconds (defaults to 500)
21763          * @type Number
21764          */
21765         frequency : 500,
21766         
21767         /**
21768          * True to animate the scroll (defaults to true)
21769          * @type Boolean
21770          */
21771         animate: true,
21772         
21773         /**
21774          * The animation duration in seconds - 
21775          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21776          * @type Number
21777          */
21778         animDuration: .4,
21779         
21780         /**
21781          * Manually trigger a cache refresh.
21782          */
21783         refreshCache : function(){
21784             for(var id in els){
21785                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21786                     els[id]._region = els[id].getRegion();
21787                 }
21788             }
21789         }
21790     };
21791 }();/*
21792  * Based on:
21793  * Ext JS Library 1.1.1
21794  * Copyright(c) 2006-2007, Ext JS, LLC.
21795  *
21796  * Originally Released Under LGPL - original licence link has changed is not relivant.
21797  *
21798  * Fork - LGPL
21799  * <script type="text/javascript">
21800  */
21801  
21802
21803 /**
21804  * @class Roo.dd.Registry
21805  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21806  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21807  * @singleton
21808  */
21809 Roo.dd.Registry = function(){
21810     var elements = {}; 
21811     var handles = {}; 
21812     var autoIdSeed = 0;
21813
21814     var getId = function(el, autogen){
21815         if(typeof el == "string"){
21816             return el;
21817         }
21818         var id = el.id;
21819         if(!id && autogen !== false){
21820             id = "roodd-" + (++autoIdSeed);
21821             el.id = id;
21822         }
21823         return id;
21824     };
21825     
21826     return {
21827     /**
21828      * Register a drag drop element
21829      * @param {String|HTMLElement} element The id or DOM node to register
21830      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21831      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21832      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21833      * populated in the data object (if applicable):
21834      * <pre>
21835 Value      Description<br />
21836 ---------  ------------------------------------------<br />
21837 handles    Array of DOM nodes that trigger dragging<br />
21838            for the element being registered<br />
21839 isHandle   True if the element passed in triggers<br />
21840            dragging itself, else false
21841 </pre>
21842      */
21843         register : function(el, data){
21844             data = data || {};
21845             if(typeof el == "string"){
21846                 el = document.getElementById(el);
21847             }
21848             data.ddel = el;
21849             elements[getId(el)] = data;
21850             if(data.isHandle !== false){
21851                 handles[data.ddel.id] = data;
21852             }
21853             if(data.handles){
21854                 var hs = data.handles;
21855                 for(var i = 0, len = hs.length; i < len; i++){
21856                         handles[getId(hs[i])] = data;
21857                 }
21858             }
21859         },
21860
21861     /**
21862      * Unregister a drag drop element
21863      * @param {String|HTMLElement}  element The id or DOM node to unregister
21864      */
21865         unregister : function(el){
21866             var id = getId(el, false);
21867             var data = elements[id];
21868             if(data){
21869                 delete elements[id];
21870                 if(data.handles){
21871                     var hs = data.handles;
21872                     for(var i = 0, len = hs.length; i < len; i++){
21873                         delete handles[getId(hs[i], false)];
21874                     }
21875                 }
21876             }
21877         },
21878
21879     /**
21880      * Returns the handle registered for a DOM Node by id
21881      * @param {String|HTMLElement} id The DOM node or id to look up
21882      * @return {Object} handle The custom handle data
21883      */
21884         getHandle : function(id){
21885             if(typeof id != "string"){ // must be element?
21886                 id = id.id;
21887             }
21888             return handles[id];
21889         },
21890
21891     /**
21892      * Returns the handle that is registered for the DOM node that is the target of the event
21893      * @param {Event} e The event
21894      * @return {Object} handle The custom handle data
21895      */
21896         getHandleFromEvent : function(e){
21897             var t = Roo.lib.Event.getTarget(e);
21898             return t ? handles[t.id] : null;
21899         },
21900
21901     /**
21902      * Returns a custom data object that is registered for a DOM node by id
21903      * @param {String|HTMLElement} id The DOM node or id to look up
21904      * @return {Object} data The custom data
21905      */
21906         getTarget : function(id){
21907             if(typeof id != "string"){ // must be element?
21908                 id = id.id;
21909             }
21910             return elements[id];
21911         },
21912
21913     /**
21914      * Returns a custom data object that is registered for the DOM node that is the target of the event
21915      * @param {Event} e The event
21916      * @return {Object} data The custom data
21917      */
21918         getTargetFromEvent : function(e){
21919             var t = Roo.lib.Event.getTarget(e);
21920             return t ? elements[t.id] || handles[t.id] : null;
21921         }
21922     };
21923 }();/*
21924  * Based on:
21925  * Ext JS Library 1.1.1
21926  * Copyright(c) 2006-2007, Ext JS, LLC.
21927  *
21928  * Originally Released Under LGPL - original licence link has changed is not relivant.
21929  *
21930  * Fork - LGPL
21931  * <script type="text/javascript">
21932  */
21933  
21934
21935 /**
21936  * @class Roo.dd.StatusProxy
21937  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21938  * default drag proxy used by all Roo.dd components.
21939  * @constructor
21940  * @param {Object} config
21941  */
21942 Roo.dd.StatusProxy = function(config){
21943     Roo.apply(this, config);
21944     this.id = this.id || Roo.id();
21945     this.el = new Roo.Layer({
21946         dh: {
21947             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21948                 {tag: "div", cls: "x-dd-drop-icon"},
21949                 {tag: "div", cls: "x-dd-drag-ghost"}
21950             ]
21951         }, 
21952         shadow: !config || config.shadow !== false
21953     });
21954     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21955     this.dropStatus = this.dropNotAllowed;
21956 };
21957
21958 Roo.dd.StatusProxy.prototype = {
21959     /**
21960      * @cfg {String} dropAllowed
21961      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21962      */
21963     dropAllowed : "x-dd-drop-ok",
21964     /**
21965      * @cfg {String} dropNotAllowed
21966      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21967      */
21968     dropNotAllowed : "x-dd-drop-nodrop",
21969
21970     /**
21971      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21972      * over the current target element.
21973      * @param {String} cssClass The css class for the new drop status indicator image
21974      */
21975     setStatus : function(cssClass){
21976         cssClass = cssClass || this.dropNotAllowed;
21977         if(this.dropStatus != cssClass){
21978             this.el.replaceClass(this.dropStatus, cssClass);
21979             this.dropStatus = cssClass;
21980         }
21981     },
21982
21983     /**
21984      * Resets the status indicator to the default dropNotAllowed value
21985      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21986      */
21987     reset : function(clearGhost){
21988         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21989         this.dropStatus = this.dropNotAllowed;
21990         if(clearGhost){
21991             this.ghost.update("");
21992         }
21993     },
21994
21995     /**
21996      * Updates the contents of the ghost element
21997      * @param {String} html The html that will replace the current innerHTML of the ghost element
21998      */
21999     update : function(html){
22000         if(typeof html == "string"){
22001             this.ghost.update(html);
22002         }else{
22003             this.ghost.update("");
22004             html.style.margin = "0";
22005             this.ghost.dom.appendChild(html);
22006         }
22007         // ensure float = none set?? cant remember why though.
22008         var el = this.ghost.dom.firstChild;
22009                 if(el){
22010                         Roo.fly(el).setStyle('float', 'none');
22011                 }
22012     },
22013     
22014     /**
22015      * Returns the underlying proxy {@link Roo.Layer}
22016      * @return {Roo.Layer} el
22017     */
22018     getEl : function(){
22019         return this.el;
22020     },
22021
22022     /**
22023      * Returns the ghost element
22024      * @return {Roo.Element} el
22025      */
22026     getGhost : function(){
22027         return this.ghost;
22028     },
22029
22030     /**
22031      * Hides the proxy
22032      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
22033      */
22034     hide : function(clear){
22035         this.el.hide();
22036         if(clear){
22037             this.reset(true);
22038         }
22039     },
22040
22041     /**
22042      * Stops the repair animation if it's currently running
22043      */
22044     stop : function(){
22045         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
22046             this.anim.stop();
22047         }
22048     },
22049
22050     /**
22051      * Displays this proxy
22052      */
22053     show : function(){
22054         this.el.show();
22055     },
22056
22057     /**
22058      * Force the Layer to sync its shadow and shim positions to the element
22059      */
22060     sync : function(){
22061         this.el.sync();
22062     },
22063
22064     /**
22065      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
22066      * invalid drop operation by the item being dragged.
22067      * @param {Array} xy The XY position of the element ([x, y])
22068      * @param {Function} callback The function to call after the repair is complete
22069      * @param {Object} scope The scope in which to execute the callback
22070      */
22071     repair : function(xy, callback, scope){
22072         this.callback = callback;
22073         this.scope = scope;
22074         if(xy && this.animRepair !== false){
22075             this.el.addClass("x-dd-drag-repair");
22076             this.el.hideUnders(true);
22077             this.anim = this.el.shift({
22078                 duration: this.repairDuration || .5,
22079                 easing: 'easeOut',
22080                 xy: xy,
22081                 stopFx: true,
22082                 callback: this.afterRepair,
22083                 scope: this
22084             });
22085         }else{
22086             this.afterRepair();
22087         }
22088     },
22089
22090     // private
22091     afterRepair : function(){
22092         this.hide(true);
22093         if(typeof this.callback == "function"){
22094             this.callback.call(this.scope || this);
22095         }
22096         this.callback = null;
22097         this.scope = null;
22098     }
22099 };/*
22100  * Based on:
22101  * Ext JS Library 1.1.1
22102  * Copyright(c) 2006-2007, Ext JS, LLC.
22103  *
22104  * Originally Released Under LGPL - original licence link has changed is not relivant.
22105  *
22106  * Fork - LGPL
22107  * <script type="text/javascript">
22108  */
22109
22110 /**
22111  * @class Roo.dd.DragSource
22112  * @extends Roo.dd.DDProxy
22113  * A simple class that provides the basic implementation needed to make any element draggable.
22114  * @constructor
22115  * @param {String/HTMLElement/Element} el The container element
22116  * @param {Object} config
22117  */
22118 Roo.dd.DragSource = function(el, config){
22119     this.el = Roo.get(el);
22120     this.dragData = {};
22121     
22122     Roo.apply(this, config);
22123     
22124     if(!this.proxy){
22125         this.proxy = new Roo.dd.StatusProxy();
22126     }
22127
22128     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
22129           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
22130     
22131     this.dragging = false;
22132 };
22133
22134 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
22135     /**
22136      * @cfg {String} dropAllowed
22137      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22138      */
22139     dropAllowed : "x-dd-drop-ok",
22140     /**
22141      * @cfg {String} dropNotAllowed
22142      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22143      */
22144     dropNotAllowed : "x-dd-drop-nodrop",
22145
22146     /**
22147      * Returns the data object associated with this drag source
22148      * @return {Object} data An object containing arbitrary data
22149      */
22150     getDragData : function(e){
22151         return this.dragData;
22152     },
22153
22154     // private
22155     onDragEnter : function(e, id){
22156         var target = Roo.dd.DragDropMgr.getDDById(id);
22157         this.cachedTarget = target;
22158         if(this.beforeDragEnter(target, e, id) !== false){
22159             if(target.isNotifyTarget){
22160                 var status = target.notifyEnter(this, e, this.dragData);
22161                 this.proxy.setStatus(status);
22162             }else{
22163                 this.proxy.setStatus(this.dropAllowed);
22164             }
22165             
22166             if(this.afterDragEnter){
22167                 /**
22168                  * An empty function by default, but provided so that you can perform a custom action
22169                  * when the dragged item enters the drop target by providing an implementation.
22170                  * @param {Roo.dd.DragDrop} target The drop target
22171                  * @param {Event} e The event object
22172                  * @param {String} id The id of the dragged element
22173                  * @method afterDragEnter
22174                  */
22175                 this.afterDragEnter(target, e, id);
22176             }
22177         }
22178     },
22179
22180     /**
22181      * An empty function by default, but provided so that you can perform a custom action
22182      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
22183      * @param {Roo.dd.DragDrop} target The drop target
22184      * @param {Event} e The event object
22185      * @param {String} id The id of the dragged element
22186      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
22187      */
22188     beforeDragEnter : function(target, e, id){
22189         return true;
22190     },
22191
22192     // private
22193     alignElWithMouse: function() {
22194         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
22195         this.proxy.sync();
22196     },
22197
22198     // private
22199     onDragOver : function(e, id){
22200         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
22201         if(this.beforeDragOver(target, e, id) !== false){
22202             if(target.isNotifyTarget){
22203                 var status = target.notifyOver(this, e, this.dragData);
22204                 this.proxy.setStatus(status);
22205             }
22206
22207             if(this.afterDragOver){
22208                 /**
22209                  * An empty function by default, but provided so that you can perform a custom action
22210                  * while the dragged item is over the drop target by providing an implementation.
22211                  * @param {Roo.dd.DragDrop} target The drop target
22212                  * @param {Event} e The event object
22213                  * @param {String} id The id of the dragged element
22214                  * @method afterDragOver
22215                  */
22216                 this.afterDragOver(target, e, id);
22217             }
22218         }
22219     },
22220
22221     /**
22222      * An empty function by default, but provided so that you can perform a custom action
22223      * while the dragged item is over the drop target and optionally cancel the onDragOver.
22224      * @param {Roo.dd.DragDrop} target The drop target
22225      * @param {Event} e The event object
22226      * @param {String} id The id of the dragged element
22227      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
22228      */
22229     beforeDragOver : function(target, e, id){
22230         return true;
22231     },
22232
22233     // private
22234     onDragOut : function(e, id){
22235         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
22236         if(this.beforeDragOut(target, e, id) !== false){
22237             if(target.isNotifyTarget){
22238                 target.notifyOut(this, e, this.dragData);
22239             }
22240             this.proxy.reset();
22241             if(this.afterDragOut){
22242                 /**
22243                  * An empty function by default, but provided so that you can perform a custom action
22244                  * after the dragged item is dragged out of the target without dropping.
22245                  * @param {Roo.dd.DragDrop} target The drop target
22246                  * @param {Event} e The event object
22247                  * @param {String} id The id of the dragged element
22248                  * @method afterDragOut
22249                  */
22250                 this.afterDragOut(target, e, id);
22251             }
22252         }
22253         this.cachedTarget = null;
22254     },
22255
22256     /**
22257      * An empty function by default, but provided so that you can perform a custom action before the dragged
22258      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
22259      * @param {Roo.dd.DragDrop} target The drop target
22260      * @param {Event} e The event object
22261      * @param {String} id The id of the dragged element
22262      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
22263      */
22264     beforeDragOut : function(target, e, id){
22265         return true;
22266     },
22267     
22268     // private
22269     onDragDrop : function(e, id){
22270         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
22271         if(this.beforeDragDrop(target, e, id) !== false){
22272             if(target.isNotifyTarget){
22273                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
22274                     this.onValidDrop(target, e, id);
22275                 }else{
22276                     this.onInvalidDrop(target, e, id);
22277                 }
22278             }else{
22279                 this.onValidDrop(target, e, id);
22280             }
22281             
22282             if(this.afterDragDrop){
22283                 /**
22284                  * An empty function by default, but provided so that you can perform a custom action
22285                  * after a valid drag drop has occurred by providing an implementation.
22286                  * @param {Roo.dd.DragDrop} target The drop target
22287                  * @param {Event} e The event object
22288                  * @param {String} id The id of the dropped element
22289                  * @method afterDragDrop
22290                  */
22291                 this.afterDragDrop(target, e, id);
22292             }
22293         }
22294         delete this.cachedTarget;
22295     },
22296
22297     /**
22298      * An empty function by default, but provided so that you can perform a custom action before the dragged
22299      * item is dropped onto the target and optionally cancel the onDragDrop.
22300      * @param {Roo.dd.DragDrop} target The drop target
22301      * @param {Event} e The event object
22302      * @param {String} id The id of the dragged element
22303      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
22304      */
22305     beforeDragDrop : function(target, e, id){
22306         return true;
22307     },
22308
22309     // private
22310     onValidDrop : function(target, e, id){
22311         this.hideProxy();
22312         if(this.afterValidDrop){
22313             /**
22314              * An empty function by default, but provided so that you can perform a custom action
22315              * after a valid drop has occurred by providing an implementation.
22316              * @param {Object} target The target DD 
22317              * @param {Event} e The event object
22318              * @param {String} id The id of the dropped element
22319              * @method afterInvalidDrop
22320              */
22321             this.afterValidDrop(target, e, id);
22322         }
22323     },
22324
22325     // private
22326     getRepairXY : function(e, data){
22327         return this.el.getXY();  
22328     },
22329
22330     // private
22331     onInvalidDrop : function(target, e, id){
22332         this.beforeInvalidDrop(target, e, id);
22333         if(this.cachedTarget){
22334             if(this.cachedTarget.isNotifyTarget){
22335                 this.cachedTarget.notifyOut(this, e, this.dragData);
22336             }
22337             this.cacheTarget = null;
22338         }
22339         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
22340
22341         if(this.afterInvalidDrop){
22342             /**
22343              * An empty function by default, but provided so that you can perform a custom action
22344              * after an invalid drop has occurred by providing an implementation.
22345              * @param {Event} e The event object
22346              * @param {String} id The id of the dropped element
22347              * @method afterInvalidDrop
22348              */
22349             this.afterInvalidDrop(e, id);
22350         }
22351     },
22352
22353     // private
22354     afterRepair : function(){
22355         if(Roo.enableFx){
22356             this.el.highlight(this.hlColor || "c3daf9");
22357         }
22358         this.dragging = false;
22359     },
22360
22361     /**
22362      * An empty function by default, but provided so that you can perform a custom action after an invalid
22363      * drop has occurred.
22364      * @param {Roo.dd.DragDrop} target The drop target
22365      * @param {Event} e The event object
22366      * @param {String} id The id of the dragged element
22367      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
22368      */
22369     beforeInvalidDrop : function(target, e, id){
22370         return true;
22371     },
22372
22373     // private
22374     handleMouseDown : function(e){
22375         if(this.dragging) {
22376             return;
22377         }
22378         var data = this.getDragData(e);
22379         if(data && this.onBeforeDrag(data, e) !== false){
22380             this.dragData = data;
22381             this.proxy.stop();
22382             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
22383         } 
22384     },
22385
22386     /**
22387      * An empty function by default, but provided so that you can perform a custom action before the initial
22388      * drag event begins and optionally cancel it.
22389      * @param {Object} data An object containing arbitrary data to be shared with drop targets
22390      * @param {Event} e The event object
22391      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
22392      */
22393     onBeforeDrag : function(data, e){
22394         return true;
22395     },
22396
22397     /**
22398      * An empty function by default, but provided so that you can perform a custom action once the initial
22399      * drag event has begun.  The drag cannot be canceled from this function.
22400      * @param {Number} x The x position of the click on the dragged object
22401      * @param {Number} y The y position of the click on the dragged object
22402      */
22403     onStartDrag : Roo.emptyFn,
22404
22405     // private - YUI override
22406     startDrag : function(x, y){
22407         this.proxy.reset();
22408         this.dragging = true;
22409         this.proxy.update("");
22410         this.onInitDrag(x, y);
22411         this.proxy.show();
22412     },
22413
22414     // private
22415     onInitDrag : function(x, y){
22416         var clone = this.el.dom.cloneNode(true);
22417         clone.id = Roo.id(); // prevent duplicate ids
22418         this.proxy.update(clone);
22419         this.onStartDrag(x, y);
22420         return true;
22421     },
22422
22423     /**
22424      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
22425      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
22426      */
22427     getProxy : function(){
22428         return this.proxy;  
22429     },
22430
22431     /**
22432      * Hides the drag source's {@link Roo.dd.StatusProxy}
22433      */
22434     hideProxy : function(){
22435         this.proxy.hide();  
22436         this.proxy.reset(true);
22437         this.dragging = false;
22438     },
22439
22440     // private
22441     triggerCacheRefresh : function(){
22442         Roo.dd.DDM.refreshCache(this.groups);
22443     },
22444
22445     // private - override to prevent hiding
22446     b4EndDrag: function(e) {
22447     },
22448
22449     // private - override to prevent moving
22450     endDrag : function(e){
22451         this.onEndDrag(this.dragData, e);
22452     },
22453
22454     // private
22455     onEndDrag : function(data, e){
22456     },
22457     
22458     // private - pin to cursor
22459     autoOffset : function(x, y) {
22460         this.setDelta(-12, -20);
22461     }    
22462 });/*
22463  * Based on:
22464  * Ext JS Library 1.1.1
22465  * Copyright(c) 2006-2007, Ext JS, LLC.
22466  *
22467  * Originally Released Under LGPL - original licence link has changed is not relivant.
22468  *
22469  * Fork - LGPL
22470  * <script type="text/javascript">
22471  */
22472
22473
22474 /**
22475  * @class Roo.dd.DropTarget
22476  * @extends Roo.dd.DDTarget
22477  * A simple class that provides the basic implementation needed to make any element a drop target that can have
22478  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
22479  * @constructor
22480  * @param {String/HTMLElement/Element} el The container element
22481  * @param {Object} config
22482  */
22483 Roo.dd.DropTarget = function(el, config){
22484     this.el = Roo.get(el);
22485     
22486     var listeners = false; ;
22487     if (config && config.listeners) {
22488         listeners= config.listeners;
22489         delete config.listeners;
22490     }
22491     Roo.apply(this, config);
22492     
22493     if(this.containerScroll){
22494         Roo.dd.ScrollManager.register(this.el);
22495     }
22496     this.addEvents( {
22497          /**
22498          * @scope Roo.dd.DropTarget
22499          */
22500          
22501          /**
22502          * @event enter
22503          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22504          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22505          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22506          * 
22507          * IMPORTANT : it should set this.overClass and this.dropAllowed
22508          * 
22509          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22510          * @param {Event} e The event
22511          * @param {Object} data An object containing arbitrary data supplied by the drag source
22512          */
22513         "enter" : true,
22514         
22515          /**
22516          * @event over
22517          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22518          * This method will be called on every mouse movement while the drag source is over the drop target.
22519          * This default implementation simply returns the dropAllowed config value.
22520          * 
22521          * IMPORTANT : it should set this.dropAllowed
22522          * 
22523          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22524          * @param {Event} e The event
22525          * @param {Object} data An object containing arbitrary data supplied by the drag source
22526          
22527          */
22528         "over" : true,
22529         /**
22530          * @event out
22531          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22532          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22533          * overClass (if any) from the drop element.
22534          * 
22535          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22536          * @param {Event} e The event
22537          * @param {Object} data An object containing arbitrary data supplied by the drag source
22538          */
22539          "out" : true,
22540          
22541         /**
22542          * @event drop
22543          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22544          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22545          * implementation that does something to process the drop event and returns true so that the drag source's
22546          * repair action does not run.
22547          * 
22548          * IMPORTANT : it should set this.success
22549          * 
22550          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22551          * @param {Event} e The event
22552          * @param {Object} data An object containing arbitrary data supplied by the drag source
22553         */
22554          "drop" : true
22555     });
22556             
22557      
22558     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22559         this.el.dom, 
22560         this.ddGroup || this.group,
22561         {
22562             isTarget: true,
22563             listeners : listeners || {} 
22564            
22565         
22566         }
22567     );
22568
22569 };
22570
22571 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22572     /**
22573      * @cfg {String} overClass
22574      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22575      */
22576      /**
22577      * @cfg {String} ddGroup
22578      * The drag drop group to handle drop events for
22579      */
22580      
22581     /**
22582      * @cfg {String} dropAllowed
22583      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22584      */
22585     dropAllowed : "x-dd-drop-ok",
22586     /**
22587      * @cfg {String} dropNotAllowed
22588      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22589      */
22590     dropNotAllowed : "x-dd-drop-nodrop",
22591     /**
22592      * @cfg {boolean} success
22593      * set this after drop listener.. 
22594      */
22595     success : false,
22596     /**
22597      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22598      * if the drop point is valid for over/enter..
22599      */
22600     valid : false,
22601     // private
22602     isTarget : true,
22603
22604     // private
22605     isNotifyTarget : true,
22606     
22607     /**
22608      * @hide
22609      */
22610     notifyEnter : function(dd, e, data)
22611     {
22612         this.valid = true;
22613         this.fireEvent('enter', dd, e, data);
22614         if(this.overClass){
22615             this.el.addClass(this.overClass);
22616         }
22617         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22618             this.valid ? this.dropAllowed : this.dropNotAllowed
22619         );
22620     },
22621
22622     /**
22623      * @hide
22624      */
22625     notifyOver : function(dd, e, data)
22626     {
22627         this.valid = true;
22628         this.fireEvent('over', dd, e, data);
22629         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22630             this.valid ? this.dropAllowed : this.dropNotAllowed
22631         );
22632     },
22633
22634     /**
22635      * @hide
22636      */
22637     notifyOut : function(dd, e, data)
22638     {
22639         this.fireEvent('out', dd, e, data);
22640         if(this.overClass){
22641             this.el.removeClass(this.overClass);
22642         }
22643     },
22644
22645     /**
22646      * @hide
22647      */
22648     notifyDrop : function(dd, e, data)
22649     {
22650         this.success = false;
22651         this.fireEvent('drop', dd, e, data);
22652         return this.success;
22653     }
22654 });/*
22655  * Based on:
22656  * Ext JS Library 1.1.1
22657  * Copyright(c) 2006-2007, Ext JS, LLC.
22658  *
22659  * Originally Released Under LGPL - original licence link has changed is not relivant.
22660  *
22661  * Fork - LGPL
22662  * <script type="text/javascript">
22663  */
22664
22665
22666 /**
22667  * @class Roo.dd.DragZone
22668  * @extends Roo.dd.DragSource
22669  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22670  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22671  * @constructor
22672  * @param {String/HTMLElement/Element} el The container element
22673  * @param {Object} config
22674  */
22675 Roo.dd.DragZone = function(el, config){
22676     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22677     if(this.containerScroll){
22678         Roo.dd.ScrollManager.register(this.el);
22679     }
22680 };
22681
22682 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22683     /**
22684      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22685      * for auto scrolling during drag operations.
22686      */
22687     /**
22688      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22689      * method after a failed drop (defaults to "c3daf9" - light blue)
22690      */
22691
22692     /**
22693      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22694      * for a valid target to drag based on the mouse down. Override this method
22695      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22696      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22697      * @param {EventObject} e The mouse down event
22698      * @return {Object} The dragData
22699      */
22700     getDragData : function(e){
22701         return Roo.dd.Registry.getHandleFromEvent(e);
22702     },
22703     
22704     /**
22705      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22706      * this.dragData.ddel
22707      * @param {Number} x The x position of the click on the dragged object
22708      * @param {Number} y The y position of the click on the dragged object
22709      * @return {Boolean} true to continue the drag, false to cancel
22710      */
22711     onInitDrag : function(x, y){
22712         this.proxy.update(this.dragData.ddel.cloneNode(true));
22713         this.onStartDrag(x, y);
22714         return true;
22715     },
22716     
22717     /**
22718      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22719      */
22720     afterRepair : function(){
22721         if(Roo.enableFx){
22722             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22723         }
22724         this.dragging = false;
22725     },
22726
22727     /**
22728      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22729      * the XY of this.dragData.ddel
22730      * @param {EventObject} e The mouse up event
22731      * @return {Array} The xy location (e.g. [100, 200])
22732      */
22733     getRepairXY : function(e){
22734         return Roo.Element.fly(this.dragData.ddel).getXY();  
22735     }
22736 });/*
22737  * Based on:
22738  * Ext JS Library 1.1.1
22739  * Copyright(c) 2006-2007, Ext JS, LLC.
22740  *
22741  * Originally Released Under LGPL - original licence link has changed is not relivant.
22742  *
22743  * Fork - LGPL
22744  * <script type="text/javascript">
22745  */
22746 /**
22747  * @class Roo.dd.DropZone
22748  * @extends Roo.dd.DropTarget
22749  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22750  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22751  * @constructor
22752  * @param {String/HTMLElement/Element} el The container element
22753  * @param {Object} config
22754  */
22755 Roo.dd.DropZone = function(el, config){
22756     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22757 };
22758
22759 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22760     /**
22761      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22762      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22763      * provide your own custom lookup.
22764      * @param {Event} e The event
22765      * @return {Object} data The custom data
22766      */
22767     getTargetFromEvent : function(e){
22768         return Roo.dd.Registry.getTargetFromEvent(e);
22769     },
22770
22771     /**
22772      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22773      * that it has registered.  This method has no default implementation and should be overridden to provide
22774      * node-specific processing if necessary.
22775      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22776      * {@link #getTargetFromEvent} for this node)
22777      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22778      * @param {Event} e The event
22779      * @param {Object} data An object containing arbitrary data supplied by the drag source
22780      */
22781     onNodeEnter : function(n, dd, e, data){
22782         
22783     },
22784
22785     /**
22786      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22787      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22788      * overridden to provide the proper feedback.
22789      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22790      * {@link #getTargetFromEvent} for this node)
22791      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22792      * @param {Event} e The event
22793      * @param {Object} data An object containing arbitrary data supplied by the drag source
22794      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22795      * underlying {@link Roo.dd.StatusProxy} can be updated
22796      */
22797     onNodeOver : function(n, dd, e, data){
22798         return this.dropAllowed;
22799     },
22800
22801     /**
22802      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22803      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22804      * node-specific processing if necessary.
22805      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22806      * {@link #getTargetFromEvent} for this node)
22807      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22808      * @param {Event} e The event
22809      * @param {Object} data An object containing arbitrary data supplied by the drag source
22810      */
22811     onNodeOut : function(n, dd, e, data){
22812         
22813     },
22814
22815     /**
22816      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22817      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22818      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22819      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22820      * {@link #getTargetFromEvent} for this node)
22821      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22822      * @param {Event} e The event
22823      * @param {Object} data An object containing arbitrary data supplied by the drag source
22824      * @return {Boolean} True if the drop was valid, else false
22825      */
22826     onNodeDrop : function(n, dd, e, data){
22827         return false;
22828     },
22829
22830     /**
22831      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22832      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22833      * it should be overridden to provide the proper feedback if necessary.
22834      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22835      * @param {Event} e The event
22836      * @param {Object} data An object containing arbitrary data supplied by the drag source
22837      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22838      * underlying {@link Roo.dd.StatusProxy} can be updated
22839      */
22840     onContainerOver : function(dd, e, data){
22841         return this.dropNotAllowed;
22842     },
22843
22844     /**
22845      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22846      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22847      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22848      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22849      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22850      * @param {Event} e The event
22851      * @param {Object} data An object containing arbitrary data supplied by the drag source
22852      * @return {Boolean} True if the drop was valid, else false
22853      */
22854     onContainerDrop : function(dd, e, data){
22855         return false;
22856     },
22857
22858     /**
22859      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22860      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22861      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22862      * you should override this method and provide a custom implementation.
22863      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22864      * @param {Event} e The event
22865      * @param {Object} data An object containing arbitrary data supplied by the drag source
22866      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22867      * underlying {@link Roo.dd.StatusProxy} can be updated
22868      */
22869     notifyEnter : function(dd, e, data){
22870         return this.dropNotAllowed;
22871     },
22872
22873     /**
22874      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22875      * This method will be called on every mouse movement while the drag source is over the drop zone.
22876      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22877      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22878      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22879      * registered node, it will call {@link #onContainerOver}.
22880      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22881      * @param {Event} e The event
22882      * @param {Object} data An object containing arbitrary data supplied by the drag source
22883      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22884      * underlying {@link Roo.dd.StatusProxy} can be updated
22885      */
22886     notifyOver : function(dd, e, data){
22887         var n = this.getTargetFromEvent(e);
22888         if(!n){ // not over valid drop target
22889             if(this.lastOverNode){
22890                 this.onNodeOut(this.lastOverNode, dd, e, data);
22891                 this.lastOverNode = null;
22892             }
22893             return this.onContainerOver(dd, e, data);
22894         }
22895         if(this.lastOverNode != n){
22896             if(this.lastOverNode){
22897                 this.onNodeOut(this.lastOverNode, dd, e, data);
22898             }
22899             this.onNodeEnter(n, dd, e, data);
22900             this.lastOverNode = n;
22901         }
22902         return this.onNodeOver(n, dd, e, data);
22903     },
22904
22905     /**
22906      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22907      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22908      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22909      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22910      * @param {Event} e The event
22911      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22912      */
22913     notifyOut : function(dd, e, data){
22914         if(this.lastOverNode){
22915             this.onNodeOut(this.lastOverNode, dd, e, data);
22916             this.lastOverNode = null;
22917         }
22918     },
22919
22920     /**
22921      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22922      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22923      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22924      * otherwise it will call {@link #onContainerDrop}.
22925      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22926      * @param {Event} e The event
22927      * @param {Object} data An object containing arbitrary data supplied by the drag source
22928      * @return {Boolean} True if the drop was valid, else false
22929      */
22930     notifyDrop : function(dd, e, data){
22931         if(this.lastOverNode){
22932             this.onNodeOut(this.lastOverNode, dd, e, data);
22933             this.lastOverNode = null;
22934         }
22935         var n = this.getTargetFromEvent(e);
22936         return n ?
22937             this.onNodeDrop(n, dd, e, data) :
22938             this.onContainerDrop(dd, e, data);
22939     },
22940
22941     // private
22942     triggerCacheRefresh : function(){
22943         Roo.dd.DDM.refreshCache(this.groups);
22944     }  
22945 });/*
22946  * Based on:
22947  * Ext JS Library 1.1.1
22948  * Copyright(c) 2006-2007, Ext JS, LLC.
22949  *
22950  * Originally Released Under LGPL - original licence link has changed is not relivant.
22951  *
22952  * Fork - LGPL
22953  * <script type="text/javascript">
22954  */
22955
22956
22957 /**
22958  * @class Roo.data.SortTypes
22959  * @singleton
22960  * Defines the default sorting (casting?) comparison functions used when sorting data.
22961  */
22962 Roo.data.SortTypes = {
22963     /**
22964      * Default sort that does nothing
22965      * @param {Mixed} s The value being converted
22966      * @return {Mixed} The comparison value
22967      */
22968     none : function(s){
22969         return s;
22970     },
22971     
22972     /**
22973      * The regular expression used to strip tags
22974      * @type {RegExp}
22975      * @property
22976      */
22977     stripTagsRE : /<\/?[^>]+>/gi,
22978     
22979     /**
22980      * Strips all HTML tags to sort on text only
22981      * @param {Mixed} s The value being converted
22982      * @return {String} The comparison value
22983      */
22984     asText : function(s){
22985         return String(s).replace(this.stripTagsRE, "");
22986     },
22987     
22988     /**
22989      * Strips all HTML tags to sort on text only - Case insensitive
22990      * @param {Mixed} s The value being converted
22991      * @return {String} The comparison value
22992      */
22993     asUCText : function(s){
22994         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22995     },
22996     
22997     /**
22998      * Case insensitive string
22999      * @param {Mixed} s The value being converted
23000      * @return {String} The comparison value
23001      */
23002     asUCString : function(s) {
23003         return String(s).toUpperCase();
23004     },
23005     
23006     /**
23007      * Date sorting
23008      * @param {Mixed} s The value being converted
23009      * @return {Number} The comparison value
23010      */
23011     asDate : function(s) {
23012         if(!s){
23013             return 0;
23014         }
23015         if(s instanceof Date){
23016             return s.getTime();
23017         }
23018         return Date.parse(String(s));
23019     },
23020     
23021     /**
23022      * Float sorting
23023      * @param {Mixed} s The value being converted
23024      * @return {Float} The comparison value
23025      */
23026     asFloat : function(s) {
23027         var val = parseFloat(String(s).replace(/,/g, ""));
23028         if(isNaN(val)) {
23029             val = 0;
23030         }
23031         return val;
23032     },
23033     
23034     /**
23035      * Integer sorting
23036      * @param {Mixed} s The value being converted
23037      * @return {Number} The comparison value
23038      */
23039     asInt : function(s) {
23040         var val = parseInt(String(s).replace(/,/g, ""));
23041         if(isNaN(val)) {
23042             val = 0;
23043         }
23044         return val;
23045     }
23046 };/*
23047  * Based on:
23048  * Ext JS Library 1.1.1
23049  * Copyright(c) 2006-2007, Ext JS, LLC.
23050  *
23051  * Originally Released Under LGPL - original licence link has changed is not relivant.
23052  *
23053  * Fork - LGPL
23054  * <script type="text/javascript">
23055  */
23056
23057 /**
23058 * @class Roo.data.Record
23059  * Instances of this class encapsulate both record <em>definition</em> information, and record
23060  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
23061  * to access Records cached in an {@link Roo.data.Store} object.<br>
23062  * <p>
23063  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
23064  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
23065  * objects.<br>
23066  * <p>
23067  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
23068  * @constructor
23069  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
23070  * {@link #create}. The parameters are the same.
23071  * @param {Array} data An associative Array of data values keyed by the field name.
23072  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
23073  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
23074  * not specified an integer id is generated.
23075  */
23076 Roo.data.Record = function(data, id){
23077     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
23078     this.data = data;
23079 };
23080
23081 /**
23082  * Generate a constructor for a specific record layout.
23083  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
23084  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
23085  * Each field definition object may contain the following properties: <ul>
23086  * <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,
23087  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
23088  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
23089  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
23090  * is being used, then this is a string containing the javascript expression to reference the data relative to 
23091  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
23092  * to the data item relative to the record element. If the mapping expression is the same as the field name,
23093  * this may be omitted.</p></li>
23094  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
23095  * <ul><li>auto (Default, implies no conversion)</li>
23096  * <li>string</li>
23097  * <li>int</li>
23098  * <li>float</li>
23099  * <li>boolean</li>
23100  * <li>date</li></ul></p></li>
23101  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
23102  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
23103  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
23104  * by the Reader into an object that will be stored in the Record. It is passed the
23105  * following parameters:<ul>
23106  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
23107  * </ul></p></li>
23108  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
23109  * </ul>
23110  * <br>usage:<br><pre><code>
23111 var TopicRecord = Roo.data.Record.create(
23112     {name: 'title', mapping: 'topic_title'},
23113     {name: 'author', mapping: 'username'},
23114     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
23115     {name: 'lastPost', mapping: 'post_time', type: 'date'},
23116     {name: 'lastPoster', mapping: 'user2'},
23117     {name: 'excerpt', mapping: 'post_text'}
23118 );
23119
23120 var myNewRecord = new TopicRecord({
23121     title: 'Do my job please',
23122     author: 'noobie',
23123     totalPosts: 1,
23124     lastPost: new Date(),
23125     lastPoster: 'Animal',
23126     excerpt: 'No way dude!'
23127 });
23128 myStore.add(myNewRecord);
23129 </code></pre>
23130  * @method create
23131  * @static
23132  */
23133 Roo.data.Record.create = function(o){
23134     var f = function(){
23135         f.superclass.constructor.apply(this, arguments);
23136     };
23137     Roo.extend(f, Roo.data.Record);
23138     var p = f.prototype;
23139     p.fields = new Roo.util.MixedCollection(false, function(field){
23140         return field.name;
23141     });
23142     for(var i = 0, len = o.length; i < len; i++){
23143         p.fields.add(new Roo.data.Field(o[i]));
23144     }
23145     f.getField = function(name){
23146         return p.fields.get(name);  
23147     };
23148     return f;
23149 };
23150
23151 Roo.data.Record.AUTO_ID = 1000;
23152 Roo.data.Record.EDIT = 'edit';
23153 Roo.data.Record.REJECT = 'reject';
23154 Roo.data.Record.COMMIT = 'commit';
23155
23156 Roo.data.Record.prototype = {
23157     /**
23158      * Readonly flag - true if this record has been modified.
23159      * @type Boolean
23160      */
23161     dirty : false,
23162     editing : false,
23163     error: null,
23164     modified: null,
23165
23166     // private
23167     join : function(store){
23168         this.store = store;
23169     },
23170
23171     /**
23172      * Set the named field to the specified value.
23173      * @param {String} name The name of the field to set.
23174      * @param {Object} value The value to set the field to.
23175      */
23176     set : function(name, value){
23177         if(this.data[name] == value){
23178             return;
23179         }
23180         this.dirty = true;
23181         if(!this.modified){
23182             this.modified = {};
23183         }
23184         if(typeof this.modified[name] == 'undefined'){
23185             this.modified[name] = this.data[name];
23186         }
23187         this.data[name] = value;
23188         if(!this.editing && this.store){
23189             this.store.afterEdit(this);
23190         }       
23191     },
23192
23193     /**
23194      * Get the value of the named field.
23195      * @param {String} name The name of the field to get the value of.
23196      * @return {Object} The value of the field.
23197      */
23198     get : function(name){
23199         return this.data[name]; 
23200     },
23201
23202     // private
23203     beginEdit : function(){
23204         this.editing = true;
23205         this.modified = {}; 
23206     },
23207
23208     // private
23209     cancelEdit : function(){
23210         this.editing = false;
23211         delete this.modified;
23212     },
23213
23214     // private
23215     endEdit : function(){
23216         this.editing = false;
23217         if(this.dirty && this.store){
23218             this.store.afterEdit(this);
23219         }
23220     },
23221
23222     /**
23223      * Usually called by the {@link Roo.data.Store} which owns the Record.
23224      * Rejects all changes made to the Record since either creation, or the last commit operation.
23225      * Modified fields are reverted to their original values.
23226      * <p>
23227      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
23228      * of reject operations.
23229      */
23230     reject : function(){
23231         var m = this.modified;
23232         for(var n in m){
23233             if(typeof m[n] != "function"){
23234                 this.data[n] = m[n];
23235             }
23236         }
23237         this.dirty = false;
23238         delete this.modified;
23239         this.editing = false;
23240         if(this.store){
23241             this.store.afterReject(this);
23242         }
23243     },
23244
23245     /**
23246      * Usually called by the {@link Roo.data.Store} which owns the Record.
23247      * Commits all changes made to the Record since either creation, or the last commit operation.
23248      * <p>
23249      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
23250      * of commit operations.
23251      */
23252     commit : function(){
23253         this.dirty = false;
23254         delete this.modified;
23255         this.editing = false;
23256         if(this.store){
23257             this.store.afterCommit(this);
23258         }
23259     },
23260
23261     // private
23262     hasError : function(){
23263         return this.error != null;
23264     },
23265
23266     // private
23267     clearError : function(){
23268         this.error = null;
23269     },
23270
23271     /**
23272      * Creates a copy of this record.
23273      * @param {String} id (optional) A new record id if you don't want to use this record's id
23274      * @return {Record}
23275      */
23276     copy : function(newId) {
23277         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
23278     }
23279 };/*
23280  * Based on:
23281  * Ext JS Library 1.1.1
23282  * Copyright(c) 2006-2007, Ext JS, LLC.
23283  *
23284  * Originally Released Under LGPL - original licence link has changed is not relivant.
23285  *
23286  * Fork - LGPL
23287  * <script type="text/javascript">
23288  */
23289
23290
23291
23292 /**
23293  * @class Roo.data.Store
23294  * @extends Roo.util.Observable
23295  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
23296  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
23297  * <p>
23298  * 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
23299  * has no knowledge of the format of the data returned by the Proxy.<br>
23300  * <p>
23301  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
23302  * instances from the data object. These records are cached and made available through accessor functions.
23303  * @constructor
23304  * Creates a new Store.
23305  * @param {Object} config A config object containing the objects needed for the Store to access data,
23306  * and read the data into Records.
23307  */
23308 Roo.data.Store = function(config){
23309     this.data = new Roo.util.MixedCollection(false);
23310     this.data.getKey = function(o){
23311         return o.id;
23312     };
23313     this.baseParams = {};
23314     // private
23315     this.paramNames = {
23316         "start" : "start",
23317         "limit" : "limit",
23318         "sort" : "sort",
23319         "dir" : "dir",
23320         "multisort" : "_multisort"
23321     };
23322
23323     if(config && config.data){
23324         this.inlineData = config.data;
23325         delete config.data;
23326     }
23327
23328     Roo.apply(this, config);
23329     
23330     if(this.reader){ // reader passed
23331         this.reader = Roo.factory(this.reader, Roo.data);
23332         this.reader.xmodule = this.xmodule || false;
23333         if(!this.recordType){
23334             this.recordType = this.reader.recordType;
23335         }
23336         if(this.reader.onMetaChange){
23337             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
23338         }
23339     }
23340
23341     if(this.recordType){
23342         this.fields = this.recordType.prototype.fields;
23343     }
23344     this.modified = [];
23345
23346     this.addEvents({
23347         /**
23348          * @event datachanged
23349          * Fires when the data cache has changed, and a widget which is using this Store
23350          * as a Record cache should refresh its view.
23351          * @param {Store} this
23352          */
23353         datachanged : true,
23354         /**
23355          * @event metachange
23356          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
23357          * @param {Store} this
23358          * @param {Object} meta The JSON metadata
23359          */
23360         metachange : true,
23361         /**
23362          * @event add
23363          * Fires when Records have been added to the Store
23364          * @param {Store} this
23365          * @param {Roo.data.Record[]} records The array of Records added
23366          * @param {Number} index The index at which the record(s) were added
23367          */
23368         add : true,
23369         /**
23370          * @event remove
23371          * Fires when a Record has been removed from the Store
23372          * @param {Store} this
23373          * @param {Roo.data.Record} record The Record that was removed
23374          * @param {Number} index The index at which the record was removed
23375          */
23376         remove : true,
23377         /**
23378          * @event update
23379          * Fires when a Record has been updated
23380          * @param {Store} this
23381          * @param {Roo.data.Record} record The Record that was updated
23382          * @param {String} operation The update operation being performed.  Value may be one of:
23383          * <pre><code>
23384  Roo.data.Record.EDIT
23385  Roo.data.Record.REJECT
23386  Roo.data.Record.COMMIT
23387          * </code></pre>
23388          */
23389         update : true,
23390         /**
23391          * @event clear
23392          * Fires when the data cache has been cleared.
23393          * @param {Store} this
23394          */
23395         clear : true,
23396         /**
23397          * @event beforeload
23398          * Fires before a request is made for a new data object.  If the beforeload handler returns false
23399          * the load action will be canceled.
23400          * @param {Store} this
23401          * @param {Object} options The loading options that were specified (see {@link #load} for details)
23402          */
23403         beforeload : true,
23404         /**
23405          * @event beforeloadadd
23406          * Fires after a new set of Records has been loaded.
23407          * @param {Store} this
23408          * @param {Roo.data.Record[]} records The Records that were loaded
23409          * @param {Object} options The loading options that were specified (see {@link #load} for details)
23410          */
23411         beforeloadadd : true,
23412         /**
23413          * @event load
23414          * Fires after a new set of Records has been loaded, before they are added to the store.
23415          * @param {Store} this
23416          * @param {Roo.data.Record[]} records The Records that were loaded
23417          * @param {Object} options The loading options that were specified (see {@link #load} for details)
23418          * @params {Object} return from reader
23419          */
23420         load : true,
23421         /**
23422          * @event loadexception
23423          * Fires if an exception occurs in the Proxy during loading.
23424          * Called with the signature of the Proxy's "loadexception" event.
23425          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
23426          * 
23427          * @param {Proxy} 
23428          * @param {Object} return from JsonData.reader() - success, totalRecords, records
23429          * @param {Object} load options 
23430          * @param {Object} jsonData from your request (normally this contains the Exception)
23431          */
23432         loadexception : true
23433     });
23434     
23435     if(this.proxy){
23436         this.proxy = Roo.factory(this.proxy, Roo.data);
23437         this.proxy.xmodule = this.xmodule || false;
23438         this.relayEvents(this.proxy,  ["loadexception"]);
23439     }
23440     this.sortToggle = {};
23441     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
23442
23443     Roo.data.Store.superclass.constructor.call(this);
23444
23445     if(this.inlineData){
23446         this.loadData(this.inlineData);
23447         delete this.inlineData;
23448     }
23449 };
23450
23451 Roo.extend(Roo.data.Store, Roo.util.Observable, {
23452      /**
23453     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
23454     * without a remote query - used by combo/forms at present.
23455     */
23456     
23457     /**
23458     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
23459     */
23460     /**
23461     * @cfg {Array} data Inline data to be loaded when the store is initialized.
23462     */
23463     /**
23464     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
23465     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
23466     */
23467     /**
23468     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
23469     * on any HTTP request
23470     */
23471     /**
23472     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
23473     */
23474     /**
23475     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
23476     */
23477     multiSort: false,
23478     /**
23479     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
23480     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
23481     */
23482     remoteSort : false,
23483
23484     /**
23485     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
23486      * loaded or when a record is removed. (defaults to false).
23487     */
23488     pruneModifiedRecords : false,
23489
23490     // private
23491     lastOptions : null,
23492
23493     /**
23494      * Add Records to the Store and fires the add event.
23495      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23496      */
23497     add : function(records){
23498         records = [].concat(records);
23499         for(var i = 0, len = records.length; i < len; i++){
23500             records[i].join(this);
23501         }
23502         var index = this.data.length;
23503         this.data.addAll(records);
23504         this.fireEvent("add", this, records, index);
23505     },
23506
23507     /**
23508      * Remove a Record from the Store and fires the remove event.
23509      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23510      */
23511     remove : function(record){
23512         var index = this.data.indexOf(record);
23513         this.data.removeAt(index);
23514  
23515         if(this.pruneModifiedRecords){
23516             this.modified.remove(record);
23517         }
23518         this.fireEvent("remove", this, record, index);
23519     },
23520
23521     /**
23522      * Remove all Records from the Store and fires the clear event.
23523      */
23524     removeAll : function(){
23525         this.data.clear();
23526         if(this.pruneModifiedRecords){
23527             this.modified = [];
23528         }
23529         this.fireEvent("clear", this);
23530     },
23531
23532     /**
23533      * Inserts Records to the Store at the given index and fires the add event.
23534      * @param {Number} index The start index at which to insert the passed Records.
23535      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23536      */
23537     insert : function(index, records){
23538         records = [].concat(records);
23539         for(var i = 0, len = records.length; i < len; i++){
23540             this.data.insert(index, records[i]);
23541             records[i].join(this);
23542         }
23543         this.fireEvent("add", this, records, index);
23544     },
23545
23546     /**
23547      * Get the index within the cache of the passed Record.
23548      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23549      * @return {Number} The index of the passed Record. Returns -1 if not found.
23550      */
23551     indexOf : function(record){
23552         return this.data.indexOf(record);
23553     },
23554
23555     /**
23556      * Get the index within the cache of the Record with the passed id.
23557      * @param {String} id The id of the Record to find.
23558      * @return {Number} The index of the Record. Returns -1 if not found.
23559      */
23560     indexOfId : function(id){
23561         return this.data.indexOfKey(id);
23562     },
23563
23564     /**
23565      * Get the Record with the specified id.
23566      * @param {String} id The id of the Record to find.
23567      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23568      */
23569     getById : function(id){
23570         return this.data.key(id);
23571     },
23572
23573     /**
23574      * Get the Record at the specified index.
23575      * @param {Number} index The index of the Record to find.
23576      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23577      */
23578     getAt : function(index){
23579         return this.data.itemAt(index);
23580     },
23581
23582     /**
23583      * Returns a range of Records between specified indices.
23584      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23585      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23586      * @return {Roo.data.Record[]} An array of Records
23587      */
23588     getRange : function(start, end){
23589         return this.data.getRange(start, end);
23590     },
23591
23592     // private
23593     storeOptions : function(o){
23594         o = Roo.apply({}, o);
23595         delete o.callback;
23596         delete o.scope;
23597         this.lastOptions = o;
23598     },
23599
23600     /**
23601      * Loads the Record cache from the configured Proxy using the configured Reader.
23602      * <p>
23603      * If using remote paging, then the first load call must specify the <em>start</em>
23604      * and <em>limit</em> properties in the options.params property to establish the initial
23605      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23606      * <p>
23607      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23608      * and this call will return before the new data has been loaded. Perform any post-processing
23609      * in a callback function, or in a "load" event handler.</strong>
23610      * <p>
23611      * @param {Object} options An object containing properties which control loading options:<ul>
23612      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23613      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23614      * passed the following arguments:<ul>
23615      * <li>r : Roo.data.Record[]</li>
23616      * <li>options: Options object from the load call</li>
23617      * <li>success: Boolean success indicator</li></ul></li>
23618      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23619      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23620      * </ul>
23621      */
23622     load : function(options){
23623         options = options || {};
23624         if(this.fireEvent("beforeload", this, options) !== false){
23625             this.storeOptions(options);
23626             var p = Roo.apply(options.params || {}, this.baseParams);
23627             // if meta was not loaded from remote source.. try requesting it.
23628             if (!this.reader.metaFromRemote) {
23629                 p._requestMeta = 1;
23630             }
23631             if(this.sortInfo && this.remoteSort){
23632                 var pn = this.paramNames;
23633                 p[pn["sort"]] = this.sortInfo.field;
23634                 p[pn["dir"]] = this.sortInfo.direction;
23635             }
23636             if (this.multiSort) {
23637                 var pn = this.paramNames;
23638                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23639             }
23640             
23641             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23642         }
23643     },
23644
23645     /**
23646      * Reloads the Record cache from the configured Proxy using the configured Reader and
23647      * the options from the last load operation performed.
23648      * @param {Object} options (optional) An object containing properties which may override the options
23649      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23650      * the most recently used options are reused).
23651      */
23652     reload : function(options){
23653         this.load(Roo.applyIf(options||{}, this.lastOptions));
23654     },
23655
23656     // private
23657     // Called as a callback by the Reader during a load operation.
23658     loadRecords : function(o, options, success){
23659         if(!o || success === false){
23660             if(success !== false){
23661                 this.fireEvent("load", this, [], options, o);
23662             }
23663             if(options.callback){
23664                 options.callback.call(options.scope || this, [], options, false);
23665             }
23666             return;
23667         }
23668         // if data returned failure - throw an exception.
23669         if (o.success === false) {
23670             // show a message if no listener is registered.
23671             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23672                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23673             }
23674             // loadmask wil be hooked into this..
23675             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23676             return;
23677         }
23678         var r = o.records, t = o.totalRecords || r.length;
23679         
23680         this.fireEvent("beforeloadadd", this, r, options, o);
23681         
23682         if(!options || options.add !== true){
23683             if(this.pruneModifiedRecords){
23684                 this.modified = [];
23685             }
23686             for(var i = 0, len = r.length; i < len; i++){
23687                 r[i].join(this);
23688             }
23689             if(this.snapshot){
23690                 this.data = this.snapshot;
23691                 delete this.snapshot;
23692             }
23693             this.data.clear();
23694             this.data.addAll(r);
23695             this.totalLength = t;
23696             this.applySort();
23697             this.fireEvent("datachanged", this);
23698         }else{
23699             this.totalLength = Math.max(t, this.data.length+r.length);
23700             this.add(r);
23701         }
23702         
23703         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23704                 
23705             var e = new Roo.data.Record({});
23706
23707             e.set(this.parent.displayField, this.parent.emptyTitle);
23708             e.set(this.parent.valueField, '');
23709
23710             this.insert(0, e);
23711         }
23712             
23713         this.fireEvent("load", this, r, options, o);
23714         if(options.callback){
23715             options.callback.call(options.scope || this, r, options, true);
23716         }
23717     },
23718
23719
23720     /**
23721      * Loads data from a passed data block. A Reader which understands the format of the data
23722      * must have been configured in the constructor.
23723      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23724      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23725      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23726      */
23727     loadData : function(o, append){
23728         var r = this.reader.readRecords(o);
23729         this.loadRecords(r, {add: append}, true);
23730     },
23731     
23732      /**
23733      * using 'cn' the nested child reader read the child array into it's child stores.
23734      * @param {Object} rec The record with a 'children array
23735      */
23736     loadDataFromChildren : function(rec)
23737     {
23738         this.loadData(this.reader.toLoadData(rec));
23739     },
23740     
23741
23742     /**
23743      * Gets the number of cached records.
23744      * <p>
23745      * <em>If using paging, this may not be the total size of the dataset. If the data object
23746      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23747      * the data set size</em>
23748      */
23749     getCount : function(){
23750         return this.data.length || 0;
23751     },
23752
23753     /**
23754      * Gets the total number of records in the dataset as returned by the server.
23755      * <p>
23756      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23757      * the dataset size</em>
23758      */
23759     getTotalCount : function(){
23760         return this.totalLength || 0;
23761     },
23762
23763     /**
23764      * Returns the sort state of the Store as an object with two properties:
23765      * <pre><code>
23766  field {String} The name of the field by which the Records are sorted
23767  direction {String} The sort order, "ASC" or "DESC"
23768      * </code></pre>
23769      */
23770     getSortState : function(){
23771         return this.sortInfo;
23772     },
23773
23774     // private
23775     applySort : function(){
23776         if(this.sortInfo && !this.remoteSort){
23777             var s = this.sortInfo, f = s.field;
23778             var st = this.fields.get(f).sortType;
23779             var fn = function(r1, r2){
23780                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23781                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23782             };
23783             this.data.sort(s.direction, fn);
23784             if(this.snapshot && this.snapshot != this.data){
23785                 this.snapshot.sort(s.direction, fn);
23786             }
23787         }
23788     },
23789
23790     /**
23791      * Sets the default sort column and order to be used by the next load operation.
23792      * @param {String} fieldName The name of the field to sort by.
23793      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23794      */
23795     setDefaultSort : function(field, dir){
23796         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23797     },
23798
23799     /**
23800      * Sort the Records.
23801      * If remote sorting is used, the sort is performed on the server, and the cache is
23802      * reloaded. If local sorting is used, the cache is sorted internally.
23803      * @param {String} fieldName The name of the field to sort by.
23804      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23805      */
23806     sort : function(fieldName, dir){
23807         var f = this.fields.get(fieldName);
23808         if(!dir){
23809             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23810             
23811             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23812                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23813             }else{
23814                 dir = f.sortDir;
23815             }
23816         }
23817         this.sortToggle[f.name] = dir;
23818         this.sortInfo = {field: f.name, direction: dir};
23819         if(!this.remoteSort){
23820             this.applySort();
23821             this.fireEvent("datachanged", this);
23822         }else{
23823             this.load(this.lastOptions);
23824         }
23825     },
23826
23827     /**
23828      * Calls the specified function for each of the Records in the cache.
23829      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23830      * Returning <em>false</em> aborts and exits the iteration.
23831      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23832      */
23833     each : function(fn, scope){
23834         this.data.each(fn, scope);
23835     },
23836
23837     /**
23838      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23839      * (e.g., during paging).
23840      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23841      */
23842     getModifiedRecords : function(){
23843         return this.modified;
23844     },
23845
23846     // private
23847     createFilterFn : function(property, value, anyMatch){
23848         if(!value.exec){ // not a regex
23849             value = String(value);
23850             if(value.length == 0){
23851                 return false;
23852             }
23853             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23854         }
23855         return function(r){
23856             return value.test(r.data[property]);
23857         };
23858     },
23859
23860     /**
23861      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23862      * @param {String} property A field on your records
23863      * @param {Number} start The record index to start at (defaults to 0)
23864      * @param {Number} end The last record index to include (defaults to length - 1)
23865      * @return {Number} The sum
23866      */
23867     sum : function(property, start, end){
23868         var rs = this.data.items, v = 0;
23869         start = start || 0;
23870         end = (end || end === 0) ? end : rs.length-1;
23871
23872         for(var i = start; i <= end; i++){
23873             v += (rs[i].data[property] || 0);
23874         }
23875         return v;
23876     },
23877
23878     /**
23879      * Filter the records by a specified property.
23880      * @param {String} field A field on your records
23881      * @param {String/RegExp} value Either a string that the field
23882      * should start with or a RegExp to test against the field
23883      * @param {Boolean} anyMatch True to match any part not just the beginning
23884      */
23885     filter : function(property, value, anyMatch){
23886         var fn = this.createFilterFn(property, value, anyMatch);
23887         return fn ? this.filterBy(fn) : this.clearFilter();
23888     },
23889
23890     /**
23891      * Filter by a function. The specified function will be called with each
23892      * record in this data source. If the function returns true the record is included,
23893      * otherwise it is filtered.
23894      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23895      * @param {Object} scope (optional) The scope of the function (defaults to this)
23896      */
23897     filterBy : function(fn, scope){
23898         this.snapshot = this.snapshot || this.data;
23899         this.data = this.queryBy(fn, scope||this);
23900         this.fireEvent("datachanged", this);
23901     },
23902
23903     /**
23904      * Query the records by a specified property.
23905      * @param {String} field A field on your records
23906      * @param {String/RegExp} value Either a string that the field
23907      * should start with or a RegExp to test against the field
23908      * @param {Boolean} anyMatch True to match any part not just the beginning
23909      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23910      */
23911     query : function(property, value, anyMatch){
23912         var fn = this.createFilterFn(property, value, anyMatch);
23913         return fn ? this.queryBy(fn) : this.data.clone();
23914     },
23915
23916     /**
23917      * Query by a function. The specified function will be called with each
23918      * record in this data source. If the function returns true the record is included
23919      * in the results.
23920      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23921      * @param {Object} scope (optional) The scope of the function (defaults to this)
23922       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23923      **/
23924     queryBy : function(fn, scope){
23925         var data = this.snapshot || this.data;
23926         return data.filterBy(fn, scope||this);
23927     },
23928
23929     /**
23930      * Collects unique values for a particular dataIndex from this store.
23931      * @param {String} dataIndex The property to collect
23932      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23933      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23934      * @return {Array} An array of the unique values
23935      **/
23936     collect : function(dataIndex, allowNull, bypassFilter){
23937         var d = (bypassFilter === true && this.snapshot) ?
23938                 this.snapshot.items : this.data.items;
23939         var v, sv, r = [], l = {};
23940         for(var i = 0, len = d.length; i < len; i++){
23941             v = d[i].data[dataIndex];
23942             sv = String(v);
23943             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23944                 l[sv] = true;
23945                 r[r.length] = v;
23946             }
23947         }
23948         return r;
23949     },
23950
23951     /**
23952      * Revert to a view of the Record cache with no filtering applied.
23953      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23954      */
23955     clearFilter : function(suppressEvent){
23956         if(this.snapshot && this.snapshot != this.data){
23957             this.data = this.snapshot;
23958             delete this.snapshot;
23959             if(suppressEvent !== true){
23960                 this.fireEvent("datachanged", this);
23961             }
23962         }
23963     },
23964
23965     // private
23966     afterEdit : function(record){
23967         if(this.modified.indexOf(record) == -1){
23968             this.modified.push(record);
23969         }
23970         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23971     },
23972     
23973     // private
23974     afterReject : function(record){
23975         this.modified.remove(record);
23976         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23977     },
23978
23979     // private
23980     afterCommit : function(record){
23981         this.modified.remove(record);
23982         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23983     },
23984
23985     /**
23986      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23987      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23988      */
23989     commitChanges : function(){
23990         var m = this.modified.slice(0);
23991         this.modified = [];
23992         for(var i = 0, len = m.length; i < len; i++){
23993             m[i].commit();
23994         }
23995     },
23996
23997     /**
23998      * Cancel outstanding changes on all changed records.
23999      */
24000     rejectChanges : function(){
24001         var m = this.modified.slice(0);
24002         this.modified = [];
24003         for(var i = 0, len = m.length; i < len; i++){
24004             m[i].reject();
24005         }
24006     },
24007
24008     onMetaChange : function(meta, rtype, o){
24009         this.recordType = rtype;
24010         this.fields = rtype.prototype.fields;
24011         delete this.snapshot;
24012         this.sortInfo = meta.sortInfo || this.sortInfo;
24013         this.modified = [];
24014         this.fireEvent('metachange', this, this.reader.meta);
24015     },
24016     
24017     moveIndex : function(data, type)
24018     {
24019         var index = this.indexOf(data);
24020         
24021         var newIndex = index + type;
24022         
24023         this.remove(data);
24024         
24025         this.insert(newIndex, data);
24026         
24027     }
24028 });/*
24029  * Based on:
24030  * Ext JS Library 1.1.1
24031  * Copyright(c) 2006-2007, Ext JS, LLC.
24032  *
24033  * Originally Released Under LGPL - original licence link has changed is not relivant.
24034  *
24035  * Fork - LGPL
24036  * <script type="text/javascript">
24037  */
24038
24039 /**
24040  * @class Roo.data.SimpleStore
24041  * @extends Roo.data.Store
24042  * Small helper class to make creating Stores from Array data easier.
24043  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
24044  * @cfg {Array} fields An array of field definition objects, or field name strings.
24045  * @cfg {Object} an existing reader (eg. copied from another store)
24046  * @cfg {Array} data The multi-dimensional array of data
24047  * @constructor
24048  * @param {Object} config
24049  */
24050 Roo.data.SimpleStore = function(config)
24051 {
24052     Roo.data.SimpleStore.superclass.constructor.call(this, {
24053         isLocal : true,
24054         reader: typeof(config.reader) != 'undefined' ? config.reader : new Roo.data.ArrayReader({
24055                 id: config.id
24056             },
24057             Roo.data.Record.create(config.fields)
24058         ),
24059         proxy : new Roo.data.MemoryProxy(config.data)
24060     });
24061     this.load();
24062 };
24063 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
24064  * Based on:
24065  * Ext JS Library 1.1.1
24066  * Copyright(c) 2006-2007, Ext JS, LLC.
24067  *
24068  * Originally Released Under LGPL - original licence link has changed is not relivant.
24069  *
24070  * Fork - LGPL
24071  * <script type="text/javascript">
24072  */
24073
24074 /**
24075 /**
24076  * @extends Roo.data.Store
24077  * @class Roo.data.JsonStore
24078  * Small helper class to make creating Stores for JSON data easier. <br/>
24079 <pre><code>
24080 var store = new Roo.data.JsonStore({
24081     url: 'get-images.php',
24082     root: 'images',
24083     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
24084 });
24085 </code></pre>
24086  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
24087  * JsonReader and HttpProxy (unless inline data is provided).</b>
24088  * @cfg {Array} fields An array of field definition objects, or field name strings.
24089  * @constructor
24090  * @param {Object} config
24091  */
24092 Roo.data.JsonStore = function(c){
24093     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
24094         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
24095         reader: new Roo.data.JsonReader(c, c.fields)
24096     }));
24097 };
24098 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
24099  * Based on:
24100  * Ext JS Library 1.1.1
24101  * Copyright(c) 2006-2007, Ext JS, LLC.
24102  *
24103  * Originally Released Under LGPL - original licence link has changed is not relivant.
24104  *
24105  * Fork - LGPL
24106  * <script type="text/javascript">
24107  */
24108
24109  
24110 Roo.data.Field = function(config){
24111     if(typeof config == "string"){
24112         config = {name: config};
24113     }
24114     Roo.apply(this, config);
24115     
24116     if(!this.type){
24117         this.type = "auto";
24118     }
24119     
24120     var st = Roo.data.SortTypes;
24121     // named sortTypes are supported, here we look them up
24122     if(typeof this.sortType == "string"){
24123         this.sortType = st[this.sortType];
24124     }
24125     
24126     // set default sortType for strings and dates
24127     if(!this.sortType){
24128         switch(this.type){
24129             case "string":
24130                 this.sortType = st.asUCString;
24131                 break;
24132             case "date":
24133                 this.sortType = st.asDate;
24134                 break;
24135             default:
24136                 this.sortType = st.none;
24137         }
24138     }
24139
24140     // define once
24141     var stripRe = /[\$,%]/g;
24142
24143     // prebuilt conversion function for this field, instead of
24144     // switching every time we're reading a value
24145     if(!this.convert){
24146         var cv, dateFormat = this.dateFormat;
24147         switch(this.type){
24148             case "":
24149             case "auto":
24150             case undefined:
24151                 cv = function(v){ return v; };
24152                 break;
24153             case "string":
24154                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
24155                 break;
24156             case "int":
24157                 cv = function(v){
24158                     return v !== undefined && v !== null && v !== '' ?
24159                            parseInt(String(v).replace(stripRe, ""), 10) : '';
24160                     };
24161                 break;
24162             case "float":
24163                 cv = function(v){
24164                     return v !== undefined && v !== null && v !== '' ?
24165                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
24166                     };
24167                 break;
24168             case "bool":
24169             case "boolean":
24170                 cv = function(v){ return v === true || v === "true" || v == 1; };
24171                 break;
24172             case "date":
24173                 cv = function(v){
24174                     if(!v){
24175                         return '';
24176                     }
24177                     if(v instanceof Date){
24178                         return v;
24179                     }
24180                     if(dateFormat){
24181                         if(dateFormat == "timestamp"){
24182                             return new Date(v*1000);
24183                         }
24184                         return Date.parseDate(v, dateFormat);
24185                     }
24186                     var parsed = Date.parse(v);
24187                     return parsed ? new Date(parsed) : null;
24188                 };
24189              break;
24190             
24191         }
24192         this.convert = cv;
24193     }
24194 };
24195
24196 Roo.data.Field.prototype = {
24197     dateFormat: null,
24198     defaultValue: "",
24199     mapping: null,
24200     sortType : null,
24201     sortDir : "ASC"
24202 };/*
24203  * Based on:
24204  * Ext JS Library 1.1.1
24205  * Copyright(c) 2006-2007, Ext JS, LLC.
24206  *
24207  * Originally Released Under LGPL - original licence link has changed is not relivant.
24208  *
24209  * Fork - LGPL
24210  * <script type="text/javascript">
24211  */
24212  
24213 // Base class for reading structured data from a data source.  This class is intended to be
24214 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
24215
24216 /**
24217  * @class Roo.data.DataReader
24218  * Base class for reading structured data from a data source.  This class is intended to be
24219  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
24220  */
24221
24222 Roo.data.DataReader = function(meta, recordType){
24223     
24224     this.meta = meta;
24225     
24226     this.recordType = recordType instanceof Array ? 
24227         Roo.data.Record.create(recordType) : recordType;
24228 };
24229
24230 Roo.data.DataReader.prototype = {
24231     
24232     
24233     readerType : 'Data',
24234      /**
24235      * Create an empty record
24236      * @param {Object} data (optional) - overlay some values
24237      * @return {Roo.data.Record} record created.
24238      */
24239     newRow :  function(d) {
24240         var da =  {};
24241         this.recordType.prototype.fields.each(function(c) {
24242             switch( c.type) {
24243                 case 'int' : da[c.name] = 0; break;
24244                 case 'date' : da[c.name] = new Date(); break;
24245                 case 'float' : da[c.name] = 0.0; break;
24246                 case 'boolean' : da[c.name] = false; break;
24247                 default : da[c.name] = ""; break;
24248             }
24249             
24250         });
24251         return new this.recordType(Roo.apply(da, d));
24252     }
24253     
24254     
24255 };/*
24256  * Based on:
24257  * Ext JS Library 1.1.1
24258  * Copyright(c) 2006-2007, Ext JS, LLC.
24259  *
24260  * Originally Released Under LGPL - original licence link has changed is not relivant.
24261  *
24262  * Fork - LGPL
24263  * <script type="text/javascript">
24264  */
24265
24266 /**
24267  * @class Roo.data.DataProxy
24268  * @extends Roo.data.Observable
24269  * This class is an abstract base class for implementations which provide retrieval of
24270  * unformatted data objects.<br>
24271  * <p>
24272  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
24273  * (of the appropriate type which knows how to parse the data object) to provide a block of
24274  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
24275  * <p>
24276  * Custom implementations must implement the load method as described in
24277  * {@link Roo.data.HttpProxy#load}.
24278  */
24279 Roo.data.DataProxy = function(){
24280     this.addEvents({
24281         /**
24282          * @event beforeload
24283          * Fires before a network request is made to retrieve a data object.
24284          * @param {Object} This DataProxy object.
24285          * @param {Object} params The params parameter to the load function.
24286          */
24287         beforeload : true,
24288         /**
24289          * @event load
24290          * Fires before the load method's callback is called.
24291          * @param {Object} This DataProxy object.
24292          * @param {Object} o The data object.
24293          * @param {Object} arg The callback argument object passed to the load function.
24294          */
24295         load : true,
24296         /**
24297          * @event loadexception
24298          * Fires if an Exception occurs during data retrieval.
24299          * @param {Object} This DataProxy object.
24300          * @param {Object} o The data object.
24301          * @param {Object} arg The callback argument object passed to the load function.
24302          * @param {Object} e The Exception.
24303          */
24304         loadexception : true
24305     });
24306     Roo.data.DataProxy.superclass.constructor.call(this);
24307 };
24308
24309 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
24310
24311     /**
24312      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
24313      */
24314 /*
24315  * Based on:
24316  * Ext JS Library 1.1.1
24317  * Copyright(c) 2006-2007, Ext JS, LLC.
24318  *
24319  * Originally Released Under LGPL - original licence link has changed is not relivant.
24320  *
24321  * Fork - LGPL
24322  * <script type="text/javascript">
24323  */
24324 /**
24325  * @class Roo.data.MemoryProxy
24326  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
24327  * to the Reader when its load method is called.
24328  * @constructor
24329  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
24330  */
24331 Roo.data.MemoryProxy = function(data){
24332     if (data.data) {
24333         data = data.data;
24334     }
24335     Roo.data.MemoryProxy.superclass.constructor.call(this);
24336     this.data = data;
24337 };
24338
24339 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
24340     
24341     /**
24342      * Load data from the requested source (in this case an in-memory
24343      * data object passed to the constructor), read the data object into
24344      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24345      * process that block using the passed callback.
24346      * @param {Object} params This parameter is not used by the MemoryProxy class.
24347      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24348      * object into a block of Roo.data.Records.
24349      * @param {Function} callback The function into which to pass the block of Roo.data.records.
24350      * The function must be passed <ul>
24351      * <li>The Record block object</li>
24352      * <li>The "arg" argument from the load function</li>
24353      * <li>A boolean success indicator</li>
24354      * </ul>
24355      * @param {Object} scope The scope in which to call the callback
24356      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24357      */
24358     load : function(params, reader, callback, scope, arg){
24359         params = params || {};
24360         var result;
24361         try {
24362             result = reader.readRecords(params.data ? params.data :this.data);
24363         }catch(e){
24364             this.fireEvent("loadexception", this, arg, null, e);
24365             callback.call(scope, null, arg, false);
24366             return;
24367         }
24368         callback.call(scope, result, arg, true);
24369     },
24370     
24371     // private
24372     update : function(params, records){
24373         
24374     }
24375 });/*
24376  * Based on:
24377  * Ext JS Library 1.1.1
24378  * Copyright(c) 2006-2007, Ext JS, LLC.
24379  *
24380  * Originally Released Under LGPL - original licence link has changed is not relivant.
24381  *
24382  * Fork - LGPL
24383  * <script type="text/javascript">
24384  */
24385 /**
24386  * @class Roo.data.HttpProxy
24387  * @extends Roo.data.DataProxy
24388  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
24389  * configured to reference a certain URL.<br><br>
24390  * <p>
24391  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
24392  * from which the running page was served.<br><br>
24393  * <p>
24394  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
24395  * <p>
24396  * Be aware that to enable the browser to parse an XML document, the server must set
24397  * the Content-Type header in the HTTP response to "text/xml".
24398  * @constructor
24399  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
24400  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
24401  * will be used to make the request.
24402  */
24403 Roo.data.HttpProxy = function(conn){
24404     Roo.data.HttpProxy.superclass.constructor.call(this);
24405     // is conn a conn config or a real conn?
24406     this.conn = conn;
24407     this.useAjax = !conn || !conn.events;
24408   
24409 };
24410
24411 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
24412     // thse are take from connection...
24413     
24414     /**
24415      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
24416      */
24417     /**
24418      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
24419      * extra parameters to each request made by this object. (defaults to undefined)
24420      */
24421     /**
24422      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
24423      *  to each request made by this object. (defaults to undefined)
24424      */
24425     /**
24426      * @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)
24427      */
24428     /**
24429      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
24430      */
24431      /**
24432      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
24433      * @type Boolean
24434      */
24435   
24436
24437     /**
24438      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
24439      * @type Boolean
24440      */
24441     /**
24442      * Return the {@link Roo.data.Connection} object being used by this Proxy.
24443      * @return {Connection} The Connection object. This object may be used to subscribe to events on
24444      * a finer-grained basis than the DataProxy events.
24445      */
24446     getConnection : function(){
24447         return this.useAjax ? Roo.Ajax : this.conn;
24448     },
24449
24450     /**
24451      * Load data from the configured {@link Roo.data.Connection}, read the data object into
24452      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
24453      * process that block using the passed callback.
24454      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24455      * for the request to the remote server.
24456      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24457      * object into a block of Roo.data.Records.
24458      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24459      * The function must be passed <ul>
24460      * <li>The Record block object</li>
24461      * <li>The "arg" argument from the load function</li>
24462      * <li>A boolean success indicator</li>
24463      * </ul>
24464      * @param {Object} scope The scope in which to call the callback
24465      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24466      */
24467     load : function(params, reader, callback, scope, arg){
24468         if(this.fireEvent("beforeload", this, params) !== false){
24469             var  o = {
24470                 params : params || {},
24471                 request: {
24472                     callback : callback,
24473                     scope : scope,
24474                     arg : arg
24475                 },
24476                 reader: reader,
24477                 callback : this.loadResponse,
24478                 scope: this
24479             };
24480             if(this.useAjax){
24481                 Roo.applyIf(o, this.conn);
24482                 if(this.activeRequest){
24483                     Roo.Ajax.abort(this.activeRequest);
24484                 }
24485                 this.activeRequest = Roo.Ajax.request(o);
24486             }else{
24487                 this.conn.request(o);
24488             }
24489         }else{
24490             callback.call(scope||this, null, arg, false);
24491         }
24492     },
24493
24494     // private
24495     loadResponse : function(o, success, response){
24496         delete this.activeRequest;
24497         if(!success){
24498             this.fireEvent("loadexception", this, o, response);
24499             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24500             return;
24501         }
24502         var result;
24503         try {
24504             result = o.reader.read(response);
24505         }catch(e){
24506             this.fireEvent("loadexception", this, o, response, e);
24507             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24508             return;
24509         }
24510         
24511         this.fireEvent("load", this, o, o.request.arg);
24512         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24513     },
24514
24515     // private
24516     update : function(dataSet){
24517
24518     },
24519
24520     // private
24521     updateResponse : function(dataSet){
24522
24523     }
24524 });/*
24525  * Based on:
24526  * Ext JS Library 1.1.1
24527  * Copyright(c) 2006-2007, Ext JS, LLC.
24528  *
24529  * Originally Released Under LGPL - original licence link has changed is not relivant.
24530  *
24531  * Fork - LGPL
24532  * <script type="text/javascript">
24533  */
24534
24535 /**
24536  * @class Roo.data.ScriptTagProxy
24537  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24538  * other than the originating domain of the running page.<br><br>
24539  * <p>
24540  * <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
24541  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24542  * <p>
24543  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24544  * source code that is used as the source inside a &lt;script> tag.<br><br>
24545  * <p>
24546  * In order for the browser to process the returned data, the server must wrap the data object
24547  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24548  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24549  * depending on whether the callback name was passed:
24550  * <p>
24551  * <pre><code>
24552 boolean scriptTag = false;
24553 String cb = request.getParameter("callback");
24554 if (cb != null) {
24555     scriptTag = true;
24556     response.setContentType("text/javascript");
24557 } else {
24558     response.setContentType("application/x-json");
24559 }
24560 Writer out = response.getWriter();
24561 if (scriptTag) {
24562     out.write(cb + "(");
24563 }
24564 out.print(dataBlock.toJsonString());
24565 if (scriptTag) {
24566     out.write(");");
24567 }
24568 </pre></code>
24569  *
24570  * @constructor
24571  * @param {Object} config A configuration object.
24572  */
24573 Roo.data.ScriptTagProxy = function(config){
24574     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24575     Roo.apply(this, config);
24576     this.head = document.getElementsByTagName("head")[0];
24577 };
24578
24579 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24580
24581 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24582     /**
24583      * @cfg {String} url The URL from which to request the data object.
24584      */
24585     /**
24586      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24587      */
24588     timeout : 30000,
24589     /**
24590      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24591      * the server the name of the callback function set up by the load call to process the returned data object.
24592      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24593      * javascript output which calls this named function passing the data object as its only parameter.
24594      */
24595     callbackParam : "callback",
24596     /**
24597      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24598      * name to the request.
24599      */
24600     nocache : true,
24601
24602     /**
24603      * Load data from the configured URL, read the data object into
24604      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24605      * process that block using the passed callback.
24606      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24607      * for the request to the remote server.
24608      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24609      * object into a block of Roo.data.Records.
24610      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24611      * The function must be passed <ul>
24612      * <li>The Record block object</li>
24613      * <li>The "arg" argument from the load function</li>
24614      * <li>A boolean success indicator</li>
24615      * </ul>
24616      * @param {Object} scope The scope in which to call the callback
24617      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24618      */
24619     load : function(params, reader, callback, scope, arg){
24620         if(this.fireEvent("beforeload", this, params) !== false){
24621
24622             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24623
24624             var url = this.url;
24625             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24626             if(this.nocache){
24627                 url += "&_dc=" + (new Date().getTime());
24628             }
24629             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24630             var trans = {
24631                 id : transId,
24632                 cb : "stcCallback"+transId,
24633                 scriptId : "stcScript"+transId,
24634                 params : params,
24635                 arg : arg,
24636                 url : url,
24637                 callback : callback,
24638                 scope : scope,
24639                 reader : reader
24640             };
24641             var conn = this;
24642
24643             window[trans.cb] = function(o){
24644                 conn.handleResponse(o, trans);
24645             };
24646
24647             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24648
24649             if(this.autoAbort !== false){
24650                 this.abort();
24651             }
24652
24653             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24654
24655             var script = document.createElement("script");
24656             script.setAttribute("src", url);
24657             script.setAttribute("type", "text/javascript");
24658             script.setAttribute("id", trans.scriptId);
24659             this.head.appendChild(script);
24660
24661             this.trans = trans;
24662         }else{
24663             callback.call(scope||this, null, arg, false);
24664         }
24665     },
24666
24667     // private
24668     isLoading : function(){
24669         return this.trans ? true : false;
24670     },
24671
24672     /**
24673      * Abort the current server request.
24674      */
24675     abort : function(){
24676         if(this.isLoading()){
24677             this.destroyTrans(this.trans);
24678         }
24679     },
24680
24681     // private
24682     destroyTrans : function(trans, isLoaded){
24683         this.head.removeChild(document.getElementById(trans.scriptId));
24684         clearTimeout(trans.timeoutId);
24685         if(isLoaded){
24686             window[trans.cb] = undefined;
24687             try{
24688                 delete window[trans.cb];
24689             }catch(e){}
24690         }else{
24691             // if hasn't been loaded, wait for load to remove it to prevent script error
24692             window[trans.cb] = function(){
24693                 window[trans.cb] = undefined;
24694                 try{
24695                     delete window[trans.cb];
24696                 }catch(e){}
24697             };
24698         }
24699     },
24700
24701     // private
24702     handleResponse : function(o, trans){
24703         this.trans = false;
24704         this.destroyTrans(trans, true);
24705         var result;
24706         try {
24707             result = trans.reader.readRecords(o);
24708         }catch(e){
24709             this.fireEvent("loadexception", this, o, trans.arg, e);
24710             trans.callback.call(trans.scope||window, null, trans.arg, false);
24711             return;
24712         }
24713         this.fireEvent("load", this, o, trans.arg);
24714         trans.callback.call(trans.scope||window, result, trans.arg, true);
24715     },
24716
24717     // private
24718     handleFailure : function(trans){
24719         this.trans = false;
24720         this.destroyTrans(trans, false);
24721         this.fireEvent("loadexception", this, null, trans.arg);
24722         trans.callback.call(trans.scope||window, null, trans.arg, false);
24723     }
24724 });/*
24725  * Based on:
24726  * Ext JS Library 1.1.1
24727  * Copyright(c) 2006-2007, Ext JS, LLC.
24728  *
24729  * Originally Released Under LGPL - original licence link has changed is not relivant.
24730  *
24731  * Fork - LGPL
24732  * <script type="text/javascript">
24733  */
24734
24735 /**
24736  * @class Roo.data.JsonReader
24737  * @extends Roo.data.DataReader
24738  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24739  * based on mappings in a provided Roo.data.Record constructor.
24740  * 
24741  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24742  * in the reply previously. 
24743  * 
24744  * <p>
24745  * Example code:
24746  * <pre><code>
24747 var RecordDef = Roo.data.Record.create([
24748     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24749     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24750 ]);
24751 var myReader = new Roo.data.JsonReader({
24752     totalProperty: "results",    // The property which contains the total dataset size (optional)
24753     root: "rows",                // The property which contains an Array of row objects
24754     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24755 }, RecordDef);
24756 </code></pre>
24757  * <p>
24758  * This would consume a JSON file like this:
24759  * <pre><code>
24760 { 'results': 2, 'rows': [
24761     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24762     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24763 }
24764 </code></pre>
24765  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24766  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24767  * paged from the remote server.
24768  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24769  * @cfg {String} root name of the property which contains the Array of row objects.
24770  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24771  * @cfg {Array} fields Array of field definition objects
24772  * @constructor
24773  * Create a new JsonReader
24774  * @param {Object} meta Metadata configuration options
24775  * @param {Object} recordType Either an Array of field definition objects,
24776  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24777  */
24778 Roo.data.JsonReader = function(meta, recordType){
24779     
24780     meta = meta || {};
24781     // set some defaults:
24782     Roo.applyIf(meta, {
24783         totalProperty: 'total',
24784         successProperty : 'success',
24785         root : 'data',
24786         id : 'id'
24787     });
24788     
24789     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24790 };
24791 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24792     
24793     readerType : 'Json',
24794     
24795     /**
24796      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24797      * Used by Store query builder to append _requestMeta to params.
24798      * 
24799      */
24800     metaFromRemote : false,
24801     /**
24802      * This method is only used by a DataProxy which has retrieved data from a remote server.
24803      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24804      * @return {Object} data A data block which is used by an Roo.data.Store object as
24805      * a cache of Roo.data.Records.
24806      */
24807     read : function(response){
24808         var json = response.responseText;
24809        
24810         var o = /* eval:var:o */ eval("("+json+")");
24811         if(!o) {
24812             throw {message: "JsonReader.read: Json object not found"};
24813         }
24814         
24815         if(o.metaData){
24816             
24817             delete this.ef;
24818             this.metaFromRemote = true;
24819             this.meta = o.metaData;
24820             this.recordType = Roo.data.Record.create(o.metaData.fields);
24821             this.onMetaChange(this.meta, this.recordType, o);
24822         }
24823         return this.readRecords(o);
24824     },
24825
24826     // private function a store will implement
24827     onMetaChange : function(meta, recordType, o){
24828
24829     },
24830
24831     /**
24832          * @ignore
24833          */
24834     simpleAccess: function(obj, subsc) {
24835         return obj[subsc];
24836     },
24837
24838         /**
24839          * @ignore
24840          */
24841     getJsonAccessor: function(){
24842         var re = /[\[\.]/;
24843         return function(expr) {
24844             try {
24845                 return(re.test(expr))
24846                     ? new Function("obj", "return obj." + expr)
24847                     : function(obj){
24848                         return obj[expr];
24849                     };
24850             } catch(e){}
24851             return Roo.emptyFn;
24852         };
24853     }(),
24854
24855     /**
24856      * Create a data block containing Roo.data.Records from an XML document.
24857      * @param {Object} o An object which contains an Array of row objects in the property specified
24858      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24859      * which contains the total size of the dataset.
24860      * @return {Object} data A data block which is used by an Roo.data.Store object as
24861      * a cache of Roo.data.Records.
24862      */
24863     readRecords : function(o){
24864         /**
24865          * After any data loads, the raw JSON data is available for further custom processing.
24866          * @type Object
24867          */
24868         this.o = o;
24869         var s = this.meta, Record = this.recordType,
24870             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24871
24872 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24873         if (!this.ef) {
24874             if(s.totalProperty) {
24875                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24876                 }
24877                 if(s.successProperty) {
24878                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24879                 }
24880                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24881                 if (s.id) {
24882                         var g = this.getJsonAccessor(s.id);
24883                         this.getId = function(rec) {
24884                                 var r = g(rec);  
24885                                 return (r === undefined || r === "") ? null : r;
24886                         };
24887                 } else {
24888                         this.getId = function(){return null;};
24889                 }
24890             this.ef = [];
24891             for(var jj = 0; jj < fl; jj++){
24892                 f = fi[jj];
24893                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24894                 this.ef[jj] = this.getJsonAccessor(map);
24895             }
24896         }
24897
24898         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24899         if(s.totalProperty){
24900             var vt = parseInt(this.getTotal(o), 10);
24901             if(!isNaN(vt)){
24902                 totalRecords = vt;
24903             }
24904         }
24905         if(s.successProperty){
24906             var vs = this.getSuccess(o);
24907             if(vs === false || vs === 'false'){
24908                 success = false;
24909             }
24910         }
24911         var records = [];
24912         for(var i = 0; i < c; i++){
24913                 var n = root[i];
24914             var values = {};
24915             var id = this.getId(n);
24916             for(var j = 0; j < fl; j++){
24917                 f = fi[j];
24918             var v = this.ef[j](n);
24919             if (!f.convert) {
24920                 Roo.log('missing convert for ' + f.name);
24921                 Roo.log(f);
24922                 continue;
24923             }
24924             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24925             }
24926             var record = new Record(values, id);
24927             record.json = n;
24928             records[i] = record;
24929         }
24930         return {
24931             raw : o,
24932             success : success,
24933             records : records,
24934             totalRecords : totalRecords
24935         };
24936     },
24937     // used when loading children.. @see loadDataFromChildren
24938     toLoadData: function(rec)
24939     {
24940         // expect rec just to be an array.. eg [a,b,c, [...] << cn ]
24941         var data = typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
24942         return { data : data, total : data.length };
24943         
24944     }
24945 });/*
24946  * Based on:
24947  * Ext JS Library 1.1.1
24948  * Copyright(c) 2006-2007, Ext JS, LLC.
24949  *
24950  * Originally Released Under LGPL - original licence link has changed is not relivant.
24951  *
24952  * Fork - LGPL
24953  * <script type="text/javascript">
24954  */
24955
24956 /**
24957  * @class Roo.data.XmlReader
24958  * @extends Roo.data.DataReader
24959  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24960  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24961  * <p>
24962  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24963  * header in the HTTP response must be set to "text/xml".</em>
24964  * <p>
24965  * Example code:
24966  * <pre><code>
24967 var RecordDef = Roo.data.Record.create([
24968    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24969    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24970 ]);
24971 var myReader = new Roo.data.XmlReader({
24972    totalRecords: "results", // The element which contains the total dataset size (optional)
24973    record: "row",           // The repeated element which contains row information
24974    id: "id"                 // The element within the row that provides an ID for the record (optional)
24975 }, RecordDef);
24976 </code></pre>
24977  * <p>
24978  * This would consume an XML file like this:
24979  * <pre><code>
24980 &lt;?xml?>
24981 &lt;dataset>
24982  &lt;results>2&lt;/results>
24983  &lt;row>
24984    &lt;id>1&lt;/id>
24985    &lt;name>Bill&lt;/name>
24986    &lt;occupation>Gardener&lt;/occupation>
24987  &lt;/row>
24988  &lt;row>
24989    &lt;id>2&lt;/id>
24990    &lt;name>Ben&lt;/name>
24991    &lt;occupation>Horticulturalist&lt;/occupation>
24992  &lt;/row>
24993 &lt;/dataset>
24994 </code></pre>
24995  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24996  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24997  * paged from the remote server.
24998  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24999  * @cfg {String} success The DomQuery path to the success attribute used by forms.
25000  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
25001  * a record identifier value.
25002  * @constructor
25003  * Create a new XmlReader
25004  * @param {Object} meta Metadata configuration options
25005  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
25006  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
25007  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
25008  */
25009 Roo.data.XmlReader = function(meta, recordType){
25010     meta = meta || {};
25011     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
25012 };
25013 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
25014     
25015     readerType : 'Xml',
25016     
25017     /**
25018      * This method is only used by a DataProxy which has retrieved data from a remote server.
25019          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
25020          * to contain a method called 'responseXML' that returns an XML document object.
25021      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
25022      * a cache of Roo.data.Records.
25023      */
25024     read : function(response){
25025         var doc = response.responseXML;
25026         if(!doc) {
25027             throw {message: "XmlReader.read: XML Document not available"};
25028         }
25029         return this.readRecords(doc);
25030     },
25031
25032     /**
25033      * Create a data block containing Roo.data.Records from an XML document.
25034          * @param {Object} doc A parsed XML document.
25035      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
25036      * a cache of Roo.data.Records.
25037      */
25038     readRecords : function(doc){
25039         /**
25040          * After any data loads/reads, the raw XML Document is available for further custom processing.
25041          * @type XMLDocument
25042          */
25043         this.xmlData = doc;
25044         var root = doc.documentElement || doc;
25045         var q = Roo.DomQuery;
25046         var recordType = this.recordType, fields = recordType.prototype.fields;
25047         var sid = this.meta.id;
25048         var totalRecords = 0, success = true;
25049         if(this.meta.totalRecords){
25050             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
25051         }
25052         
25053         if(this.meta.success){
25054             var sv = q.selectValue(this.meta.success, root, true);
25055             success = sv !== false && sv !== 'false';
25056         }
25057         var records = [];
25058         var ns = q.select(this.meta.record, root);
25059         for(var i = 0, len = ns.length; i < len; i++) {
25060                 var n = ns[i];
25061                 var values = {};
25062                 var id = sid ? q.selectValue(sid, n) : undefined;
25063                 for(var j = 0, jlen = fields.length; j < jlen; j++){
25064                     var f = fields.items[j];
25065                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
25066                     v = f.convert(v);
25067                     values[f.name] = v;
25068                 }
25069                 var record = new recordType(values, id);
25070                 record.node = n;
25071                 records[records.length] = record;
25072             }
25073
25074             return {
25075                 success : success,
25076                 records : records,
25077                 totalRecords : totalRecords || records.length
25078             };
25079     }
25080 });/*
25081  * Based on:
25082  * Ext JS Library 1.1.1
25083  * Copyright(c) 2006-2007, Ext JS, LLC.
25084  *
25085  * Originally Released Under LGPL - original licence link has changed is not relivant.
25086  *
25087  * Fork - LGPL
25088  * <script type="text/javascript">
25089  */
25090
25091 /**
25092  * @class Roo.data.ArrayReader
25093  * @extends Roo.data.DataReader
25094  * Data reader class to create an Array of Roo.data.Record objects from an Array.
25095  * Each element of that Array represents a row of data fields. The
25096  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
25097  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
25098  * <p>
25099  * Example code:.
25100  * <pre><code>
25101 var RecordDef = Roo.data.Record.create([
25102     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
25103     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
25104 ]);
25105 var myReader = new Roo.data.ArrayReader({
25106     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
25107 }, RecordDef);
25108 </code></pre>
25109  * <p>
25110  * This would consume an Array like this:
25111  * <pre><code>
25112 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
25113   </code></pre>
25114  
25115  * @constructor
25116  * Create a new JsonReader
25117  * @param {Object} meta Metadata configuration options.
25118  * @param {Object|Array} recordType Either an Array of field definition objects
25119  * 
25120  * @cfg {Array} fields Array of field definition objects
25121  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
25122  * as specified to {@link Roo.data.Record#create},
25123  * or an {@link Roo.data.Record} object
25124  *
25125  * 
25126  * created using {@link Roo.data.Record#create}.
25127  */
25128 Roo.data.ArrayReader = function(meta, recordType)
25129 {    
25130     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
25131 };
25132
25133 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
25134     
25135       /**
25136      * Create a data block containing Roo.data.Records from an XML document.
25137      * @param {Object} o An Array of row objects which represents the dataset.
25138      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
25139      * a cache of Roo.data.Records.
25140      */
25141     readRecords : function(o)
25142     {
25143         var sid = this.meta ? this.meta.id : null;
25144         var recordType = this.recordType, fields = recordType.prototype.fields;
25145         var records = [];
25146         var root = o;
25147         for(var i = 0; i < root.length; i++){
25148                 var n = root[i];
25149             var values = {};
25150             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
25151             for(var j = 0, jlen = fields.length; j < jlen; j++){
25152                 var f = fields.items[j];
25153                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
25154                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
25155                 v = f.convert(v);
25156                 values[f.name] = v;
25157             }
25158             var record = new recordType(values, id);
25159             record.json = n;
25160             records[records.length] = record;
25161         }
25162         return {
25163             records : records,
25164             totalRecords : records.length
25165         };
25166     },
25167     // used when loading children.. @see loadDataFromChildren
25168     toLoadData: function(rec)
25169     {
25170         // expect rec just to be an array.. eg [a,b,c, [...] << cn ]
25171         return typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
25172         
25173     }
25174     
25175     
25176 });/*
25177  * Based on:
25178  * Ext JS Library 1.1.1
25179  * Copyright(c) 2006-2007, Ext JS, LLC.
25180  *
25181  * Originally Released Under LGPL - original licence link has changed is not relivant.
25182  *
25183  * Fork - LGPL
25184  * <script type="text/javascript">
25185  */
25186
25187
25188 /**
25189  * @class Roo.data.Tree
25190  * @extends Roo.util.Observable
25191  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
25192  * in the tree have most standard DOM functionality.
25193  * @constructor
25194  * @param {Node} root (optional) The root node
25195  */
25196 Roo.data.Tree = function(root){
25197    this.nodeHash = {};
25198    /**
25199     * The root node for this tree
25200     * @type Node
25201     */
25202    this.root = null;
25203    if(root){
25204        this.setRootNode(root);
25205    }
25206    this.addEvents({
25207        /**
25208         * @event append
25209         * Fires when a new child node is appended to a node in this tree.
25210         * @param {Tree} tree The owner tree
25211         * @param {Node} parent The parent node
25212         * @param {Node} node The newly appended node
25213         * @param {Number} index The index of the newly appended node
25214         */
25215        "append" : true,
25216        /**
25217         * @event remove
25218         * Fires when a child node is removed from a node in this tree.
25219         * @param {Tree} tree The owner tree
25220         * @param {Node} parent The parent node
25221         * @param {Node} node The child node removed
25222         */
25223        "remove" : true,
25224        /**
25225         * @event move
25226         * Fires when a node is moved to a new location in the tree
25227         * @param {Tree} tree The owner tree
25228         * @param {Node} node The node moved
25229         * @param {Node} oldParent The old parent of this node
25230         * @param {Node} newParent The new parent of this node
25231         * @param {Number} index The index it was moved to
25232         */
25233        "move" : true,
25234        /**
25235         * @event insert
25236         * Fires when a new child node is inserted in a node in this tree.
25237         * @param {Tree} tree The owner tree
25238         * @param {Node} parent The parent node
25239         * @param {Node} node The child node inserted
25240         * @param {Node} refNode The child node the node was inserted before
25241         */
25242        "insert" : true,
25243        /**
25244         * @event beforeappend
25245         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
25246         * @param {Tree} tree The owner tree
25247         * @param {Node} parent The parent node
25248         * @param {Node} node The child node to be appended
25249         */
25250        "beforeappend" : true,
25251        /**
25252         * @event beforeremove
25253         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
25254         * @param {Tree} tree The owner tree
25255         * @param {Node} parent The parent node
25256         * @param {Node} node The child node to be removed
25257         */
25258        "beforeremove" : true,
25259        /**
25260         * @event beforemove
25261         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
25262         * @param {Tree} tree The owner tree
25263         * @param {Node} node The node being moved
25264         * @param {Node} oldParent The parent of the node
25265         * @param {Node} newParent The new parent the node is moving to
25266         * @param {Number} index The index it is being moved to
25267         */
25268        "beforemove" : true,
25269        /**
25270         * @event beforeinsert
25271         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
25272         * @param {Tree} tree The owner tree
25273         * @param {Node} parent The parent node
25274         * @param {Node} node The child node to be inserted
25275         * @param {Node} refNode The child node the node is being inserted before
25276         */
25277        "beforeinsert" : true
25278    });
25279
25280     Roo.data.Tree.superclass.constructor.call(this);
25281 };
25282
25283 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
25284     pathSeparator: "/",
25285
25286     proxyNodeEvent : function(){
25287         return this.fireEvent.apply(this, arguments);
25288     },
25289
25290     /**
25291      * Returns the root node for this tree.
25292      * @return {Node}
25293      */
25294     getRootNode : function(){
25295         return this.root;
25296     },
25297
25298     /**
25299      * Sets the root node for this tree.
25300      * @param {Node} node
25301      * @return {Node}
25302      */
25303     setRootNode : function(node){
25304         this.root = node;
25305         node.ownerTree = this;
25306         node.isRoot = true;
25307         this.registerNode(node);
25308         return node;
25309     },
25310
25311     /**
25312      * Gets a node in this tree by its id.
25313      * @param {String} id
25314      * @return {Node}
25315      */
25316     getNodeById : function(id){
25317         return this.nodeHash[id];
25318     },
25319
25320     registerNode : function(node){
25321         this.nodeHash[node.id] = node;
25322     },
25323
25324     unregisterNode : function(node){
25325         delete this.nodeHash[node.id];
25326     },
25327
25328     toString : function(){
25329         return "[Tree"+(this.id?" "+this.id:"")+"]";
25330     }
25331 });
25332
25333 /**
25334  * @class Roo.data.Node
25335  * @extends Roo.util.Observable
25336  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
25337  * @cfg {String} id The id for this node. If one is not specified, one is generated.
25338  * @constructor
25339  * @param {Object} attributes The attributes/config for the node
25340  */
25341 Roo.data.Node = function(attributes){
25342     /**
25343      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
25344      * @type {Object}
25345      */
25346     this.attributes = attributes || {};
25347     this.leaf = this.attributes.leaf;
25348     /**
25349      * The node id. @type String
25350      */
25351     this.id = this.attributes.id;
25352     if(!this.id){
25353         this.id = Roo.id(null, "ynode-");
25354         this.attributes.id = this.id;
25355     }
25356      
25357     
25358     /**
25359      * All child nodes of this node. @type Array
25360      */
25361     this.childNodes = [];
25362     if(!this.childNodes.indexOf){ // indexOf is a must
25363         this.childNodes.indexOf = function(o){
25364             for(var i = 0, len = this.length; i < len; i++){
25365                 if(this[i] == o) {
25366                     return i;
25367                 }
25368             }
25369             return -1;
25370         };
25371     }
25372     /**
25373      * The parent node for this node. @type Node
25374      */
25375     this.parentNode = null;
25376     /**
25377      * The first direct child node of this node, or null if this node has no child nodes. @type Node
25378      */
25379     this.firstChild = null;
25380     /**
25381      * The last direct child node of this node, or null if this node has no child nodes. @type Node
25382      */
25383     this.lastChild = null;
25384     /**
25385      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
25386      */
25387     this.previousSibling = null;
25388     /**
25389      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
25390      */
25391     this.nextSibling = null;
25392
25393     this.addEvents({
25394        /**
25395         * @event append
25396         * Fires when a new child node is appended
25397         * @param {Tree} tree The owner tree
25398         * @param {Node} this This node
25399         * @param {Node} node The newly appended node
25400         * @param {Number} index The index of the newly appended node
25401         */
25402        "append" : true,
25403        /**
25404         * @event remove
25405         * Fires when a child node is removed
25406         * @param {Tree} tree The owner tree
25407         * @param {Node} this This node
25408         * @param {Node} node The removed node
25409         */
25410        "remove" : true,
25411        /**
25412         * @event move
25413         * Fires when this node is moved to a new location in the tree
25414         * @param {Tree} tree The owner tree
25415         * @param {Node} this This node
25416         * @param {Node} oldParent The old parent of this node
25417         * @param {Node} newParent The new parent of this node
25418         * @param {Number} index The index it was moved to
25419         */
25420        "move" : true,
25421        /**
25422         * @event insert
25423         * Fires when a new child node is inserted.
25424         * @param {Tree} tree The owner tree
25425         * @param {Node} this This node
25426         * @param {Node} node The child node inserted
25427         * @param {Node} refNode The child node the node was inserted before
25428         */
25429        "insert" : true,
25430        /**
25431         * @event beforeappend
25432         * Fires before a new child is appended, return false to cancel the append.
25433         * @param {Tree} tree The owner tree
25434         * @param {Node} this This node
25435         * @param {Node} node The child node to be appended
25436         */
25437        "beforeappend" : true,
25438        /**
25439         * @event beforeremove
25440         * Fires before a child is removed, return false to cancel the remove.
25441         * @param {Tree} tree The owner tree
25442         * @param {Node} this This node
25443         * @param {Node} node The child node to be removed
25444         */
25445        "beforeremove" : true,
25446        /**
25447         * @event beforemove
25448         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
25449         * @param {Tree} tree The owner tree
25450         * @param {Node} this This node
25451         * @param {Node} oldParent The parent of this node
25452         * @param {Node} newParent The new parent this node is moving to
25453         * @param {Number} index The index it is being moved to
25454         */
25455        "beforemove" : true,
25456        /**
25457         * @event beforeinsert
25458         * Fires before a new child is inserted, return false to cancel the insert.
25459         * @param {Tree} tree The owner tree
25460         * @param {Node} this This node
25461         * @param {Node} node The child node to be inserted
25462         * @param {Node} refNode The child node the node is being inserted before
25463         */
25464        "beforeinsert" : true
25465    });
25466     this.listeners = this.attributes.listeners;
25467     Roo.data.Node.superclass.constructor.call(this);
25468 };
25469
25470 Roo.extend(Roo.data.Node, Roo.util.Observable, {
25471     fireEvent : function(evtName){
25472         // first do standard event for this node
25473         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
25474             return false;
25475         }
25476         // then bubble it up to the tree if the event wasn't cancelled
25477         var ot = this.getOwnerTree();
25478         if(ot){
25479             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
25480                 return false;
25481             }
25482         }
25483         return true;
25484     },
25485
25486     /**
25487      * Returns true if this node is a leaf
25488      * @return {Boolean}
25489      */
25490     isLeaf : function(){
25491         return this.leaf === true;
25492     },
25493
25494     // private
25495     setFirstChild : function(node){
25496         this.firstChild = node;
25497     },
25498
25499     //private
25500     setLastChild : function(node){
25501         this.lastChild = node;
25502     },
25503
25504
25505     /**
25506      * Returns true if this node is the last child of its parent
25507      * @return {Boolean}
25508      */
25509     isLast : function(){
25510        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25511     },
25512
25513     /**
25514      * Returns true if this node is the first child of its parent
25515      * @return {Boolean}
25516      */
25517     isFirst : function(){
25518        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25519     },
25520
25521     hasChildNodes : function(){
25522         return !this.isLeaf() && this.childNodes.length > 0;
25523     },
25524
25525     /**
25526      * Insert node(s) as the last child node of this node.
25527      * @param {Node/Array} node The node or Array of nodes to append
25528      * @return {Node} The appended node if single append, or null if an array was passed
25529      */
25530     appendChild : function(node){
25531         var multi = false;
25532         if(node instanceof Array){
25533             multi = node;
25534         }else if(arguments.length > 1){
25535             multi = arguments;
25536         }
25537         
25538         // if passed an array or multiple args do them one by one
25539         if(multi){
25540             for(var i = 0, len = multi.length; i < len; i++) {
25541                 this.appendChild(multi[i]);
25542             }
25543         }else{
25544             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25545                 return false;
25546             }
25547             var index = this.childNodes.length;
25548             var oldParent = node.parentNode;
25549             // it's a move, make sure we move it cleanly
25550             if(oldParent){
25551                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25552                     return false;
25553                 }
25554                 oldParent.removeChild(node);
25555             }
25556             
25557             index = this.childNodes.length;
25558             if(index == 0){
25559                 this.setFirstChild(node);
25560             }
25561             this.childNodes.push(node);
25562             node.parentNode = this;
25563             var ps = this.childNodes[index-1];
25564             if(ps){
25565                 node.previousSibling = ps;
25566                 ps.nextSibling = node;
25567             }else{
25568                 node.previousSibling = null;
25569             }
25570             node.nextSibling = null;
25571             this.setLastChild(node);
25572             node.setOwnerTree(this.getOwnerTree());
25573             this.fireEvent("append", this.ownerTree, this, node, index);
25574             if(this.ownerTree) {
25575                 this.ownerTree.fireEvent("appendnode", this, node, index);
25576             }
25577             if(oldParent){
25578                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25579             }
25580             return node;
25581         }
25582     },
25583
25584     /**
25585      * Removes a child node from this node.
25586      * @param {Node} node The node to remove
25587      * @return {Node} The removed node
25588      */
25589     removeChild : function(node){
25590         var index = this.childNodes.indexOf(node);
25591         if(index == -1){
25592             return false;
25593         }
25594         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25595             return false;
25596         }
25597
25598         // remove it from childNodes collection
25599         this.childNodes.splice(index, 1);
25600
25601         // update siblings
25602         if(node.previousSibling){
25603             node.previousSibling.nextSibling = node.nextSibling;
25604         }
25605         if(node.nextSibling){
25606             node.nextSibling.previousSibling = node.previousSibling;
25607         }
25608
25609         // update child refs
25610         if(this.firstChild == node){
25611             this.setFirstChild(node.nextSibling);
25612         }
25613         if(this.lastChild == node){
25614             this.setLastChild(node.previousSibling);
25615         }
25616
25617         node.setOwnerTree(null);
25618         // clear any references from the node
25619         node.parentNode = null;
25620         node.previousSibling = null;
25621         node.nextSibling = null;
25622         this.fireEvent("remove", this.ownerTree, this, node);
25623         return node;
25624     },
25625
25626     /**
25627      * Inserts the first node before the second node in this nodes childNodes collection.
25628      * @param {Node} node The node to insert
25629      * @param {Node} refNode The node to insert before (if null the node is appended)
25630      * @return {Node} The inserted node
25631      */
25632     insertBefore : function(node, refNode){
25633         if(!refNode){ // like standard Dom, refNode can be null for append
25634             return this.appendChild(node);
25635         }
25636         // nothing to do
25637         if(node == refNode){
25638             return false;
25639         }
25640
25641         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25642             return false;
25643         }
25644         var index = this.childNodes.indexOf(refNode);
25645         var oldParent = node.parentNode;
25646         var refIndex = index;
25647
25648         // when moving internally, indexes will change after remove
25649         if(oldParent == this && this.childNodes.indexOf(node) < index){
25650             refIndex--;
25651         }
25652
25653         // it's a move, make sure we move it cleanly
25654         if(oldParent){
25655             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25656                 return false;
25657             }
25658             oldParent.removeChild(node);
25659         }
25660         if(refIndex == 0){
25661             this.setFirstChild(node);
25662         }
25663         this.childNodes.splice(refIndex, 0, node);
25664         node.parentNode = this;
25665         var ps = this.childNodes[refIndex-1];
25666         if(ps){
25667             node.previousSibling = ps;
25668             ps.nextSibling = node;
25669         }else{
25670             node.previousSibling = null;
25671         }
25672         node.nextSibling = refNode;
25673         refNode.previousSibling = node;
25674         node.setOwnerTree(this.getOwnerTree());
25675         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25676         if(oldParent){
25677             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25678         }
25679         return node;
25680     },
25681
25682     /**
25683      * Returns the child node at the specified index.
25684      * @param {Number} index
25685      * @return {Node}
25686      */
25687     item : function(index){
25688         return this.childNodes[index];
25689     },
25690
25691     /**
25692      * Replaces one child node in this node with another.
25693      * @param {Node} newChild The replacement node
25694      * @param {Node} oldChild The node to replace
25695      * @return {Node} The replaced node
25696      */
25697     replaceChild : function(newChild, oldChild){
25698         this.insertBefore(newChild, oldChild);
25699         this.removeChild(oldChild);
25700         return oldChild;
25701     },
25702
25703     /**
25704      * Returns the index of a child node
25705      * @param {Node} node
25706      * @return {Number} The index of the node or -1 if it was not found
25707      */
25708     indexOf : function(child){
25709         return this.childNodes.indexOf(child);
25710     },
25711
25712     /**
25713      * Returns the tree this node is in.
25714      * @return {Tree}
25715      */
25716     getOwnerTree : function(){
25717         // if it doesn't have one, look for one
25718         if(!this.ownerTree){
25719             var p = this;
25720             while(p){
25721                 if(p.ownerTree){
25722                     this.ownerTree = p.ownerTree;
25723                     break;
25724                 }
25725                 p = p.parentNode;
25726             }
25727         }
25728         return this.ownerTree;
25729     },
25730
25731     /**
25732      * Returns depth of this node (the root node has a depth of 0)
25733      * @return {Number}
25734      */
25735     getDepth : function(){
25736         var depth = 0;
25737         var p = this;
25738         while(p.parentNode){
25739             ++depth;
25740             p = p.parentNode;
25741         }
25742         return depth;
25743     },
25744
25745     // private
25746     setOwnerTree : function(tree){
25747         // if it's move, we need to update everyone
25748         if(tree != this.ownerTree){
25749             if(this.ownerTree){
25750                 this.ownerTree.unregisterNode(this);
25751             }
25752             this.ownerTree = tree;
25753             var cs = this.childNodes;
25754             for(var i = 0, len = cs.length; i < len; i++) {
25755                 cs[i].setOwnerTree(tree);
25756             }
25757             if(tree){
25758                 tree.registerNode(this);
25759             }
25760         }
25761     },
25762
25763     /**
25764      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25765      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25766      * @return {String} The path
25767      */
25768     getPath : function(attr){
25769         attr = attr || "id";
25770         var p = this.parentNode;
25771         var b = [this.attributes[attr]];
25772         while(p){
25773             b.unshift(p.attributes[attr]);
25774             p = p.parentNode;
25775         }
25776         var sep = this.getOwnerTree().pathSeparator;
25777         return sep + b.join(sep);
25778     },
25779
25780     /**
25781      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25782      * function call will be the scope provided or the current node. The arguments to the function
25783      * will be the args provided or the current node. If the function returns false at any point,
25784      * the bubble is stopped.
25785      * @param {Function} fn The function to call
25786      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25787      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25788      */
25789     bubble : function(fn, scope, args){
25790         var p = this;
25791         while(p){
25792             if(fn.call(scope || p, args || p) === false){
25793                 break;
25794             }
25795             p = p.parentNode;
25796         }
25797     },
25798
25799     /**
25800      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25801      * function call will be the scope provided or the current node. The arguments to the function
25802      * will be the args provided or the current node. If the function returns false at any point,
25803      * the cascade is stopped on that branch.
25804      * @param {Function} fn The function to call
25805      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25806      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25807      */
25808     cascade : function(fn, scope, args){
25809         if(fn.call(scope || this, args || this) !== false){
25810             var cs = this.childNodes;
25811             for(var i = 0, len = cs.length; i < len; i++) {
25812                 cs[i].cascade(fn, scope, args);
25813             }
25814         }
25815     },
25816
25817     /**
25818      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25819      * function call will be the scope provided or the current node. The arguments to the function
25820      * will be the args provided or the current node. If the function returns false at any point,
25821      * the iteration stops.
25822      * @param {Function} fn The function to call
25823      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25824      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25825      */
25826     eachChild : function(fn, scope, args){
25827         var cs = this.childNodes;
25828         for(var i = 0, len = cs.length; i < len; i++) {
25829                 if(fn.call(scope || this, args || cs[i]) === false){
25830                     break;
25831                 }
25832         }
25833     },
25834
25835     /**
25836      * Finds the first child that has the attribute with the specified value.
25837      * @param {String} attribute The attribute name
25838      * @param {Mixed} value The value to search for
25839      * @return {Node} The found child or null if none was found
25840      */
25841     findChild : function(attribute, value){
25842         var cs = this.childNodes;
25843         for(var i = 0, len = cs.length; i < len; i++) {
25844                 if(cs[i].attributes[attribute] == value){
25845                     return cs[i];
25846                 }
25847         }
25848         return null;
25849     },
25850
25851     /**
25852      * Finds the first child by a custom function. The child matches if the function passed
25853      * returns true.
25854      * @param {Function} fn
25855      * @param {Object} scope (optional)
25856      * @return {Node} The found child or null if none was found
25857      */
25858     findChildBy : function(fn, scope){
25859         var cs = this.childNodes;
25860         for(var i = 0, len = cs.length; i < len; i++) {
25861                 if(fn.call(scope||cs[i], cs[i]) === true){
25862                     return cs[i];
25863                 }
25864         }
25865         return null;
25866     },
25867
25868     /**
25869      * Sorts this nodes children using the supplied sort function
25870      * @param {Function} fn
25871      * @param {Object} scope (optional)
25872      */
25873     sort : function(fn, scope){
25874         var cs = this.childNodes;
25875         var len = cs.length;
25876         if(len > 0){
25877             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25878             cs.sort(sortFn);
25879             for(var i = 0; i < len; i++){
25880                 var n = cs[i];
25881                 n.previousSibling = cs[i-1];
25882                 n.nextSibling = cs[i+1];
25883                 if(i == 0){
25884                     this.setFirstChild(n);
25885                 }
25886                 if(i == len-1){
25887                     this.setLastChild(n);
25888                 }
25889             }
25890         }
25891     },
25892
25893     /**
25894      * Returns true if this node is an ancestor (at any point) of the passed node.
25895      * @param {Node} node
25896      * @return {Boolean}
25897      */
25898     contains : function(node){
25899         return node.isAncestor(this);
25900     },
25901
25902     /**
25903      * Returns true if the passed node is an ancestor (at any point) of this node.
25904      * @param {Node} node
25905      * @return {Boolean}
25906      */
25907     isAncestor : function(node){
25908         var p = this.parentNode;
25909         while(p){
25910             if(p == node){
25911                 return true;
25912             }
25913             p = p.parentNode;
25914         }
25915         return false;
25916     },
25917
25918     toString : function(){
25919         return "[Node"+(this.id?" "+this.id:"")+"]";
25920     }
25921 });/*
25922  * Based on:
25923  * Ext JS Library 1.1.1
25924  * Copyright(c) 2006-2007, Ext JS, LLC.
25925  *
25926  * Originally Released Under LGPL - original licence link has changed is not relivant.
25927  *
25928  * Fork - LGPL
25929  * <script type="text/javascript">
25930  */
25931
25932
25933 /**
25934  * @class Roo.Shadow
25935  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25936  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25937  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25938  * @constructor
25939  * Create a new Shadow
25940  * @param {Object} config The config object
25941  */
25942 Roo.Shadow = function(config){
25943     Roo.apply(this, config);
25944     if(typeof this.mode != "string"){
25945         this.mode = this.defaultMode;
25946     }
25947     var o = this.offset, a = {h: 0};
25948     var rad = Math.floor(this.offset/2);
25949     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25950         case "drop":
25951             a.w = 0;
25952             a.l = a.t = o;
25953             a.t -= 1;
25954             if(Roo.isIE){
25955                 a.l -= this.offset + rad;
25956                 a.t -= this.offset + rad;
25957                 a.w -= rad;
25958                 a.h -= rad;
25959                 a.t += 1;
25960             }
25961         break;
25962         case "sides":
25963             a.w = (o*2);
25964             a.l = -o;
25965             a.t = o-1;
25966             if(Roo.isIE){
25967                 a.l -= (this.offset - rad);
25968                 a.t -= this.offset + rad;
25969                 a.l += 1;
25970                 a.w -= (this.offset - rad)*2;
25971                 a.w -= rad + 1;
25972                 a.h -= 1;
25973             }
25974         break;
25975         case "frame":
25976             a.w = a.h = (o*2);
25977             a.l = a.t = -o;
25978             a.t += 1;
25979             a.h -= 2;
25980             if(Roo.isIE){
25981                 a.l -= (this.offset - rad);
25982                 a.t -= (this.offset - rad);
25983                 a.l += 1;
25984                 a.w -= (this.offset + rad + 1);
25985                 a.h -= (this.offset + rad);
25986                 a.h += 1;
25987             }
25988         break;
25989     };
25990
25991     this.adjusts = a;
25992 };
25993
25994 Roo.Shadow.prototype = {
25995     /**
25996      * @cfg {String} mode
25997      * The shadow display mode.  Supports the following options:<br />
25998      * sides: Shadow displays on both sides and bottom only<br />
25999      * frame: Shadow displays equally on all four sides<br />
26000      * drop: Traditional bottom-right drop shadow (default)
26001      */
26002     /**
26003      * @cfg {String} offset
26004      * The number of pixels to offset the shadow from the element (defaults to 4)
26005      */
26006     offset: 4,
26007
26008     // private
26009     defaultMode: "drop",
26010
26011     /**
26012      * Displays the shadow under the target element
26013      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
26014      */
26015     show : function(target){
26016         target = Roo.get(target);
26017         if(!this.el){
26018             this.el = Roo.Shadow.Pool.pull();
26019             if(this.el.dom.nextSibling != target.dom){
26020                 this.el.insertBefore(target);
26021             }
26022         }
26023         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
26024         if(Roo.isIE){
26025             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
26026         }
26027         this.realign(
26028             target.getLeft(true),
26029             target.getTop(true),
26030             target.getWidth(),
26031             target.getHeight()
26032         );
26033         this.el.dom.style.display = "block";
26034     },
26035
26036     /**
26037      * Returns true if the shadow is visible, else false
26038      */
26039     isVisible : function(){
26040         return this.el ? true : false;  
26041     },
26042
26043     /**
26044      * Direct alignment when values are already available. Show must be called at least once before
26045      * calling this method to ensure it is initialized.
26046      * @param {Number} left The target element left position
26047      * @param {Number} top The target element top position
26048      * @param {Number} width The target element width
26049      * @param {Number} height The target element height
26050      */
26051     realign : function(l, t, w, h){
26052         if(!this.el){
26053             return;
26054         }
26055         var a = this.adjusts, d = this.el.dom, s = d.style;
26056         var iea = 0;
26057         s.left = (l+a.l)+"px";
26058         s.top = (t+a.t)+"px";
26059         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26060  
26061         if(s.width != sws || s.height != shs){
26062             s.width = sws;
26063             s.height = shs;
26064             if(!Roo.isIE){
26065                 var cn = d.childNodes;
26066                 var sww = Math.max(0, (sw-12))+"px";
26067                 cn[0].childNodes[1].style.width = sww;
26068                 cn[1].childNodes[1].style.width = sww;
26069                 cn[2].childNodes[1].style.width = sww;
26070                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26071             }
26072         }
26073     },
26074
26075     /**
26076      * Hides this shadow
26077      */
26078     hide : function(){
26079         if(this.el){
26080             this.el.dom.style.display = "none";
26081             Roo.Shadow.Pool.push(this.el);
26082             delete this.el;
26083         }
26084     },
26085
26086     /**
26087      * Adjust the z-index of this shadow
26088      * @param {Number} zindex The new z-index
26089      */
26090     setZIndex : function(z){
26091         this.zIndex = z;
26092         if(this.el){
26093             this.el.setStyle("z-index", z);
26094         }
26095     }
26096 };
26097
26098 // Private utility class that manages the internal Shadow cache
26099 Roo.Shadow.Pool = function(){
26100     var p = [];
26101     var markup = Roo.isIE ?
26102                  '<div class="x-ie-shadow"></div>' :
26103                  '<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>';
26104     return {
26105         pull : function(){
26106             var sh = p.shift();
26107             if(!sh){
26108                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26109                 sh.autoBoxAdjust = false;
26110             }
26111             return sh;
26112         },
26113
26114         push : function(sh){
26115             p.push(sh);
26116         }
26117     };
26118 }();/*
26119  * Based on:
26120  * Ext JS Library 1.1.1
26121  * Copyright(c) 2006-2007, Ext JS, LLC.
26122  *
26123  * Originally Released Under LGPL - original licence link has changed is not relivant.
26124  *
26125  * Fork - LGPL
26126  * <script type="text/javascript">
26127  */
26128
26129
26130 /**
26131  * @class Roo.SplitBar
26132  * @extends Roo.util.Observable
26133  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26134  * <br><br>
26135  * Usage:
26136  * <pre><code>
26137 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26138                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26139 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26140 split.minSize = 100;
26141 split.maxSize = 600;
26142 split.animate = true;
26143 split.on('moved', splitterMoved);
26144 </code></pre>
26145  * @constructor
26146  * Create a new SplitBar
26147  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26148  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26149  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26150  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26151                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26152                         position of the SplitBar).
26153  */
26154 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26155     
26156     /** @private */
26157     this.el = Roo.get(dragElement, true);
26158     this.el.dom.unselectable = "on";
26159     /** @private */
26160     this.resizingEl = Roo.get(resizingElement, true);
26161
26162     /**
26163      * @private
26164      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26165      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26166      * @type Number
26167      */
26168     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26169     
26170     /**
26171      * The minimum size of the resizing element. (Defaults to 0)
26172      * @type Number
26173      */
26174     this.minSize = 0;
26175     
26176     /**
26177      * The maximum size of the resizing element. (Defaults to 2000)
26178      * @type Number
26179      */
26180     this.maxSize = 2000;
26181     
26182     /**
26183      * Whether to animate the transition to the new size
26184      * @type Boolean
26185      */
26186     this.animate = false;
26187     
26188     /**
26189      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26190      * @type Boolean
26191      */
26192     this.useShim = false;
26193     
26194     /** @private */
26195     this.shim = null;
26196     
26197     if(!existingProxy){
26198         /** @private */
26199         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26200     }else{
26201         this.proxy = Roo.get(existingProxy).dom;
26202     }
26203     /** @private */
26204     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26205     
26206     /** @private */
26207     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26208     
26209     /** @private */
26210     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26211     
26212     /** @private */
26213     this.dragSpecs = {};
26214     
26215     /**
26216      * @private The adapter to use to positon and resize elements
26217      */
26218     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26219     this.adapter.init(this);
26220     
26221     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26222         /** @private */
26223         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26224         this.el.addClass("x-splitbar-h");
26225     }else{
26226         /** @private */
26227         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26228         this.el.addClass("x-splitbar-v");
26229     }
26230     
26231     this.addEvents({
26232         /**
26233          * @event resize
26234          * Fires when the splitter is moved (alias for {@link #event-moved})
26235          * @param {Roo.SplitBar} this
26236          * @param {Number} newSize the new width or height
26237          */
26238         "resize" : true,
26239         /**
26240          * @event moved
26241          * Fires when the splitter is moved
26242          * @param {Roo.SplitBar} this
26243          * @param {Number} newSize the new width or height
26244          */
26245         "moved" : true,
26246         /**
26247          * @event beforeresize
26248          * Fires before the splitter is dragged
26249          * @param {Roo.SplitBar} this
26250          */
26251         "beforeresize" : true,
26252
26253         "beforeapply" : true
26254     });
26255
26256     Roo.util.Observable.call(this);
26257 };
26258
26259 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26260     onStartProxyDrag : function(x, y){
26261         this.fireEvent("beforeresize", this);
26262         if(!this.overlay){
26263             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26264             o.unselectable();
26265             o.enableDisplayMode("block");
26266             // all splitbars share the same overlay
26267             Roo.SplitBar.prototype.overlay = o;
26268         }
26269         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26270         this.overlay.show();
26271         Roo.get(this.proxy).setDisplayed("block");
26272         var size = this.adapter.getElementSize(this);
26273         this.activeMinSize = this.getMinimumSize();;
26274         this.activeMaxSize = this.getMaximumSize();;
26275         var c1 = size - this.activeMinSize;
26276         var c2 = Math.max(this.activeMaxSize - size, 0);
26277         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26278             this.dd.resetConstraints();
26279             this.dd.setXConstraint(
26280                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26281                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26282             );
26283             this.dd.setYConstraint(0, 0);
26284         }else{
26285             this.dd.resetConstraints();
26286             this.dd.setXConstraint(0, 0);
26287             this.dd.setYConstraint(
26288                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26289                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26290             );
26291          }
26292         this.dragSpecs.startSize = size;
26293         this.dragSpecs.startPoint = [x, y];
26294         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26295     },
26296     
26297     /** 
26298      * @private Called after the drag operation by the DDProxy
26299      */
26300     onEndProxyDrag : function(e){
26301         Roo.get(this.proxy).setDisplayed(false);
26302         var endPoint = Roo.lib.Event.getXY(e);
26303         if(this.overlay){
26304             this.overlay.hide();
26305         }
26306         var newSize;
26307         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26308             newSize = this.dragSpecs.startSize + 
26309                 (this.placement == Roo.SplitBar.LEFT ?
26310                     endPoint[0] - this.dragSpecs.startPoint[0] :
26311                     this.dragSpecs.startPoint[0] - endPoint[0]
26312                 );
26313         }else{
26314             newSize = this.dragSpecs.startSize + 
26315                 (this.placement == Roo.SplitBar.TOP ?
26316                     endPoint[1] - this.dragSpecs.startPoint[1] :
26317                     this.dragSpecs.startPoint[1] - endPoint[1]
26318                 );
26319         }
26320         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26321         if(newSize != this.dragSpecs.startSize){
26322             if(this.fireEvent('beforeapply', this, newSize) !== false){
26323                 this.adapter.setElementSize(this, newSize);
26324                 this.fireEvent("moved", this, newSize);
26325                 this.fireEvent("resize", this, newSize);
26326             }
26327         }
26328     },
26329     
26330     /**
26331      * Get the adapter this SplitBar uses
26332      * @return The adapter object
26333      */
26334     getAdapter : function(){
26335         return this.adapter;
26336     },
26337     
26338     /**
26339      * Set the adapter this SplitBar uses
26340      * @param {Object} adapter A SplitBar adapter object
26341      */
26342     setAdapter : function(adapter){
26343         this.adapter = adapter;
26344         this.adapter.init(this);
26345     },
26346     
26347     /**
26348      * Gets the minimum size for the resizing element
26349      * @return {Number} The minimum size
26350      */
26351     getMinimumSize : function(){
26352         return this.minSize;
26353     },
26354     
26355     /**
26356      * Sets the minimum size for the resizing element
26357      * @param {Number} minSize The minimum size
26358      */
26359     setMinimumSize : function(minSize){
26360         this.minSize = minSize;
26361     },
26362     
26363     /**
26364      * Gets the maximum size for the resizing element
26365      * @return {Number} The maximum size
26366      */
26367     getMaximumSize : function(){
26368         return this.maxSize;
26369     },
26370     
26371     /**
26372      * Sets the maximum size for the resizing element
26373      * @param {Number} maxSize The maximum size
26374      */
26375     setMaximumSize : function(maxSize){
26376         this.maxSize = maxSize;
26377     },
26378     
26379     /**
26380      * Sets the initialize size for the resizing element
26381      * @param {Number} size The initial size
26382      */
26383     setCurrentSize : function(size){
26384         var oldAnimate = this.animate;
26385         this.animate = false;
26386         this.adapter.setElementSize(this, size);
26387         this.animate = oldAnimate;
26388     },
26389     
26390     /**
26391      * Destroy this splitbar. 
26392      * @param {Boolean} removeEl True to remove the element
26393      */
26394     destroy : function(removeEl){
26395         if(this.shim){
26396             this.shim.remove();
26397         }
26398         this.dd.unreg();
26399         this.proxy.parentNode.removeChild(this.proxy);
26400         if(removeEl){
26401             this.el.remove();
26402         }
26403     }
26404 });
26405
26406 /**
26407  * @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.
26408  */
26409 Roo.SplitBar.createProxy = function(dir){
26410     var proxy = new Roo.Element(document.createElement("div"));
26411     proxy.unselectable();
26412     var cls = 'x-splitbar-proxy';
26413     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26414     document.body.appendChild(proxy.dom);
26415     return proxy.dom;
26416 };
26417
26418 /** 
26419  * @class Roo.SplitBar.BasicLayoutAdapter
26420  * Default Adapter. It assumes the splitter and resizing element are not positioned
26421  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26422  */
26423 Roo.SplitBar.BasicLayoutAdapter = function(){
26424 };
26425
26426 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26427     // do nothing for now
26428     init : function(s){
26429     
26430     },
26431     /**
26432      * Called before drag operations to get the current size of the resizing element. 
26433      * @param {Roo.SplitBar} s The SplitBar using this adapter
26434      */
26435      getElementSize : function(s){
26436         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26437             return s.resizingEl.getWidth();
26438         }else{
26439             return s.resizingEl.getHeight();
26440         }
26441     },
26442     
26443     /**
26444      * Called after drag operations to set the size of the resizing element.
26445      * @param {Roo.SplitBar} s The SplitBar using this adapter
26446      * @param {Number} newSize The new size to set
26447      * @param {Function} onComplete A function to be invoked when resizing is complete
26448      */
26449     setElementSize : function(s, newSize, onComplete){
26450         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26451             if(!s.animate){
26452                 s.resizingEl.setWidth(newSize);
26453                 if(onComplete){
26454                     onComplete(s, newSize);
26455                 }
26456             }else{
26457                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26458             }
26459         }else{
26460             
26461             if(!s.animate){
26462                 s.resizingEl.setHeight(newSize);
26463                 if(onComplete){
26464                     onComplete(s, newSize);
26465                 }
26466             }else{
26467                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26468             }
26469         }
26470     }
26471 };
26472
26473 /** 
26474  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26475  * @extends Roo.SplitBar.BasicLayoutAdapter
26476  * Adapter that  moves the splitter element to align with the resized sizing element. 
26477  * Used with an absolute positioned SplitBar.
26478  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26479  * document.body, make sure you assign an id to the body element.
26480  */
26481 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26482     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26483     this.container = Roo.get(container);
26484 };
26485
26486 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26487     init : function(s){
26488         this.basic.init(s);
26489     },
26490     
26491     getElementSize : function(s){
26492         return this.basic.getElementSize(s);
26493     },
26494     
26495     setElementSize : function(s, newSize, onComplete){
26496         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26497     },
26498     
26499     moveSplitter : function(s){
26500         var yes = Roo.SplitBar;
26501         switch(s.placement){
26502             case yes.LEFT:
26503                 s.el.setX(s.resizingEl.getRight());
26504                 break;
26505             case yes.RIGHT:
26506                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26507                 break;
26508             case yes.TOP:
26509                 s.el.setY(s.resizingEl.getBottom());
26510                 break;
26511             case yes.BOTTOM:
26512                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26513                 break;
26514         }
26515     }
26516 };
26517
26518 /**
26519  * Orientation constant - Create a vertical SplitBar
26520  * @static
26521  * @type Number
26522  */
26523 Roo.SplitBar.VERTICAL = 1;
26524
26525 /**
26526  * Orientation constant - Create a horizontal SplitBar
26527  * @static
26528  * @type Number
26529  */
26530 Roo.SplitBar.HORIZONTAL = 2;
26531
26532 /**
26533  * Placement constant - The resizing element is to the left of the splitter element
26534  * @static
26535  * @type Number
26536  */
26537 Roo.SplitBar.LEFT = 1;
26538
26539 /**
26540  * Placement constant - The resizing element is to the right of the splitter element
26541  * @static
26542  * @type Number
26543  */
26544 Roo.SplitBar.RIGHT = 2;
26545
26546 /**
26547  * Placement constant - The resizing element is positioned above the splitter element
26548  * @static
26549  * @type Number
26550  */
26551 Roo.SplitBar.TOP = 3;
26552
26553 /**
26554  * Placement constant - The resizing element is positioned under splitter element
26555  * @static
26556  * @type Number
26557  */
26558 Roo.SplitBar.BOTTOM = 4;
26559 /*
26560  * Based on:
26561  * Ext JS Library 1.1.1
26562  * Copyright(c) 2006-2007, Ext JS, LLC.
26563  *
26564  * Originally Released Under LGPL - original licence link has changed is not relivant.
26565  *
26566  * Fork - LGPL
26567  * <script type="text/javascript">
26568  */
26569
26570 /**
26571  * @class Roo.View
26572  * @extends Roo.util.Observable
26573  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26574  * This class also supports single and multi selection modes. <br>
26575  * Create a data model bound view:
26576  <pre><code>
26577  var store = new Roo.data.Store(...);
26578
26579  var view = new Roo.View({
26580     el : "my-element",
26581     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26582  
26583     singleSelect: true,
26584     selectedClass: "ydataview-selected",
26585     store: store
26586  });
26587
26588  // listen for node click?
26589  view.on("click", function(vw, index, node, e){
26590  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26591  });
26592
26593  // load XML data
26594  dataModel.load("foobar.xml");
26595  </code></pre>
26596  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26597  * <br><br>
26598  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26599  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26600  * 
26601  * Note: old style constructor is still suported (container, template, config)
26602  * 
26603  * @constructor
26604  * Create a new View
26605  * @param {Object} config The config object
26606  * 
26607  */
26608 Roo.View = function(config, depreciated_tpl, depreciated_config){
26609     
26610     this.parent = false;
26611     
26612     if (typeof(depreciated_tpl) == 'undefined') {
26613         // new way.. - universal constructor.
26614         Roo.apply(this, config);
26615         this.el  = Roo.get(this.el);
26616     } else {
26617         // old format..
26618         this.el  = Roo.get(config);
26619         this.tpl = depreciated_tpl;
26620         Roo.apply(this, depreciated_config);
26621     }
26622     this.wrapEl  = this.el.wrap().wrap();
26623     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26624     
26625     
26626     if(typeof(this.tpl) == "string"){
26627         this.tpl = new Roo.Template(this.tpl);
26628     } else {
26629         // support xtype ctors..
26630         this.tpl = new Roo.factory(this.tpl, Roo);
26631     }
26632     
26633     
26634     this.tpl.compile();
26635     
26636     /** @private */
26637     this.addEvents({
26638         /**
26639          * @event beforeclick
26640          * Fires before a click is processed. Returns false to cancel the default action.
26641          * @param {Roo.View} this
26642          * @param {Number} index The index of the target node
26643          * @param {HTMLElement} node The target node
26644          * @param {Roo.EventObject} e The raw event object
26645          */
26646             "beforeclick" : true,
26647         /**
26648          * @event click
26649          * Fires when a template node is clicked.
26650          * @param {Roo.View} this
26651          * @param {Number} index The index of the target node
26652          * @param {HTMLElement} node The target node
26653          * @param {Roo.EventObject} e The raw event object
26654          */
26655             "click" : true,
26656         /**
26657          * @event dblclick
26658          * Fires when a template node is double clicked.
26659          * @param {Roo.View} this
26660          * @param {Number} index The index of the target node
26661          * @param {HTMLElement} node The target node
26662          * @param {Roo.EventObject} e The raw event object
26663          */
26664             "dblclick" : true,
26665         /**
26666          * @event contextmenu
26667          * Fires when a template node is right clicked.
26668          * @param {Roo.View} this
26669          * @param {Number} index The index of the target node
26670          * @param {HTMLElement} node The target node
26671          * @param {Roo.EventObject} e The raw event object
26672          */
26673             "contextmenu" : true,
26674         /**
26675          * @event selectionchange
26676          * Fires when the selected nodes change.
26677          * @param {Roo.View} this
26678          * @param {Array} selections Array of the selected nodes
26679          */
26680             "selectionchange" : true,
26681     
26682         /**
26683          * @event beforeselect
26684          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26685          * @param {Roo.View} this
26686          * @param {HTMLElement} node The node to be selected
26687          * @param {Array} selections Array of currently selected nodes
26688          */
26689             "beforeselect" : true,
26690         /**
26691          * @event preparedata
26692          * Fires on every row to render, to allow you to change the data.
26693          * @param {Roo.View} this
26694          * @param {Object} data to be rendered (change this)
26695          */
26696           "preparedata" : true
26697           
26698           
26699         });
26700
26701
26702
26703     this.el.on({
26704         "click": this.onClick,
26705         "dblclick": this.onDblClick,
26706         "contextmenu": this.onContextMenu,
26707         scope:this
26708     });
26709
26710     this.selections = [];
26711     this.nodes = [];
26712     this.cmp = new Roo.CompositeElementLite([]);
26713     if(this.store){
26714         this.store = Roo.factory(this.store, Roo.data);
26715         this.setStore(this.store, true);
26716     }
26717     
26718     if ( this.footer && this.footer.xtype) {
26719            
26720          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26721         
26722         this.footer.dataSource = this.store;
26723         this.footer.container = fctr;
26724         this.footer = Roo.factory(this.footer, Roo);
26725         fctr.insertFirst(this.el);
26726         
26727         // this is a bit insane - as the paging toolbar seems to detach the el..
26728 //        dom.parentNode.parentNode.parentNode
26729          // they get detached?
26730     }
26731     
26732     
26733     Roo.View.superclass.constructor.call(this);
26734     
26735     
26736 };
26737
26738 Roo.extend(Roo.View, Roo.util.Observable, {
26739     
26740      /**
26741      * @cfg {Roo.data.Store} store Data store to load data from.
26742      */
26743     store : false,
26744     
26745     /**
26746      * @cfg {String|Roo.Element} el The container element.
26747      */
26748     el : '',
26749     
26750     /**
26751      * @cfg {String|Roo.Template} tpl The template used by this View 
26752      */
26753     tpl : false,
26754     /**
26755      * @cfg {String} dataName the named area of the template to use as the data area
26756      *                          Works with domtemplates roo-name="name"
26757      */
26758     dataName: false,
26759     /**
26760      * @cfg {String} selectedClass The css class to add to selected nodes
26761      */
26762     selectedClass : "x-view-selected",
26763      /**
26764      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26765      */
26766     emptyText : "",
26767     
26768     /**
26769      * @cfg {String} text to display on mask (default Loading)
26770      */
26771     mask : false,
26772     /**
26773      * @cfg {Boolean} multiSelect Allow multiple selection
26774      */
26775     multiSelect : false,
26776     /**
26777      * @cfg {Boolean} singleSelect Allow single selection
26778      */
26779     singleSelect:  false,
26780     
26781     /**
26782      * @cfg {Boolean} toggleSelect - selecting 
26783      */
26784     toggleSelect : false,
26785     
26786     /**
26787      * @cfg {Boolean} tickable - selecting 
26788      */
26789     tickable : false,
26790     
26791     /**
26792      * Returns the element this view is bound to.
26793      * @return {Roo.Element}
26794      */
26795     getEl : function(){
26796         return this.wrapEl;
26797     },
26798     
26799     
26800
26801     /**
26802      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26803      */
26804     refresh : function(){
26805         //Roo.log('refresh');
26806         var t = this.tpl;
26807         
26808         // if we are using something like 'domtemplate', then
26809         // the what gets used is:
26810         // t.applySubtemplate(NAME, data, wrapping data..)
26811         // the outer template then get' applied with
26812         //     the store 'extra data'
26813         // and the body get's added to the
26814         //      roo-name="data" node?
26815         //      <span class='roo-tpl-{name}'></span> ?????
26816         
26817         
26818         
26819         this.clearSelections();
26820         this.el.update("");
26821         var html = [];
26822         var records = this.store.getRange();
26823         if(records.length < 1) {
26824             
26825             // is this valid??  = should it render a template??
26826             
26827             this.el.update(this.emptyText);
26828             return;
26829         }
26830         var el = this.el;
26831         if (this.dataName) {
26832             this.el.update(t.apply(this.store.meta)); //????
26833             el = this.el.child('.roo-tpl-' + this.dataName);
26834         }
26835         
26836         for(var i = 0, len = records.length; i < len; i++){
26837             var data = this.prepareData(records[i].data, i, records[i]);
26838             this.fireEvent("preparedata", this, data, i, records[i]);
26839             
26840             var d = Roo.apply({}, data);
26841             
26842             if(this.tickable){
26843                 Roo.apply(d, {'roo-id' : Roo.id()});
26844                 
26845                 var _this = this;
26846             
26847                 Roo.each(this.parent.item, function(item){
26848                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26849                         return;
26850                     }
26851                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26852                 });
26853             }
26854             
26855             html[html.length] = Roo.util.Format.trim(
26856                 this.dataName ?
26857                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26858                     t.apply(d)
26859             );
26860         }
26861         
26862         
26863         
26864         el.update(html.join(""));
26865         this.nodes = el.dom.childNodes;
26866         this.updateIndexes(0);
26867     },
26868     
26869
26870     /**
26871      * Function to override to reformat the data that is sent to
26872      * the template for each node.
26873      * DEPRICATED - use the preparedata event handler.
26874      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26875      * a JSON object for an UpdateManager bound view).
26876      */
26877     prepareData : function(data, index, record)
26878     {
26879         this.fireEvent("preparedata", this, data, index, record);
26880         return data;
26881     },
26882
26883     onUpdate : function(ds, record){
26884         // Roo.log('on update');   
26885         this.clearSelections();
26886         var index = this.store.indexOf(record);
26887         var n = this.nodes[index];
26888         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26889         n.parentNode.removeChild(n);
26890         this.updateIndexes(index, index);
26891     },
26892
26893     
26894     
26895 // --------- FIXME     
26896     onAdd : function(ds, records, index)
26897     {
26898         //Roo.log(['on Add', ds, records, index] );        
26899         this.clearSelections();
26900         if(this.nodes.length == 0){
26901             this.refresh();
26902             return;
26903         }
26904         var n = this.nodes[index];
26905         for(var i = 0, len = records.length; i < len; i++){
26906             var d = this.prepareData(records[i].data, i, records[i]);
26907             if(n){
26908                 this.tpl.insertBefore(n, d);
26909             }else{
26910                 
26911                 this.tpl.append(this.el, d);
26912             }
26913         }
26914         this.updateIndexes(index);
26915     },
26916
26917     onRemove : function(ds, record, index){
26918        // Roo.log('onRemove');
26919         this.clearSelections();
26920         var el = this.dataName  ?
26921             this.el.child('.roo-tpl-' + this.dataName) :
26922             this.el; 
26923         
26924         el.dom.removeChild(this.nodes[index]);
26925         this.updateIndexes(index);
26926     },
26927
26928     /**
26929      * Refresh an individual node.
26930      * @param {Number} index
26931      */
26932     refreshNode : function(index){
26933         this.onUpdate(this.store, this.store.getAt(index));
26934     },
26935
26936     updateIndexes : function(startIndex, endIndex){
26937         var ns = this.nodes;
26938         startIndex = startIndex || 0;
26939         endIndex = endIndex || ns.length - 1;
26940         for(var i = startIndex; i <= endIndex; i++){
26941             ns[i].nodeIndex = i;
26942         }
26943     },
26944
26945     /**
26946      * Changes the data store this view uses and refresh the view.
26947      * @param {Store} store
26948      */
26949     setStore : function(store, initial){
26950         if(!initial && this.store){
26951             this.store.un("datachanged", this.refresh);
26952             this.store.un("add", this.onAdd);
26953             this.store.un("remove", this.onRemove);
26954             this.store.un("update", this.onUpdate);
26955             this.store.un("clear", this.refresh);
26956             this.store.un("beforeload", this.onBeforeLoad);
26957             this.store.un("load", this.onLoad);
26958             this.store.un("loadexception", this.onLoad);
26959         }
26960         if(store){
26961           
26962             store.on("datachanged", this.refresh, this);
26963             store.on("add", this.onAdd, this);
26964             store.on("remove", this.onRemove, this);
26965             store.on("update", this.onUpdate, this);
26966             store.on("clear", this.refresh, this);
26967             store.on("beforeload", this.onBeforeLoad, this);
26968             store.on("load", this.onLoad, this);
26969             store.on("loadexception", this.onLoad, this);
26970         }
26971         
26972         if(store){
26973             this.refresh();
26974         }
26975     },
26976     /**
26977      * onbeforeLoad - masks the loading area.
26978      *
26979      */
26980     onBeforeLoad : function(store,opts)
26981     {
26982          //Roo.log('onBeforeLoad');   
26983         if (!opts.add) {
26984             this.el.update("");
26985         }
26986         this.el.mask(this.mask ? this.mask : "Loading" ); 
26987     },
26988     onLoad : function ()
26989     {
26990         this.el.unmask();
26991     },
26992     
26993
26994     /**
26995      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26996      * @param {HTMLElement} node
26997      * @return {HTMLElement} The template node
26998      */
26999     findItemFromChild : function(node){
27000         var el = this.dataName  ?
27001             this.el.child('.roo-tpl-' + this.dataName,true) :
27002             this.el.dom; 
27003         
27004         if(!node || node.parentNode == el){
27005                     return node;
27006             }
27007             var p = node.parentNode;
27008             while(p && p != el){
27009             if(p.parentNode == el){
27010                 return p;
27011             }
27012             p = p.parentNode;
27013         }
27014             return null;
27015     },
27016
27017     /** @ignore */
27018     onClick : function(e){
27019         var item = this.findItemFromChild(e.getTarget());
27020         if(item){
27021             var index = this.indexOf(item);
27022             if(this.onItemClick(item, index, e) !== false){
27023                 this.fireEvent("click", this, index, item, e);
27024             }
27025         }else{
27026             this.clearSelections();
27027         }
27028     },
27029
27030     /** @ignore */
27031     onContextMenu : function(e){
27032         var item = this.findItemFromChild(e.getTarget());
27033         if(item){
27034             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
27035         }
27036     },
27037
27038     /** @ignore */
27039     onDblClick : function(e){
27040         var item = this.findItemFromChild(e.getTarget());
27041         if(item){
27042             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
27043         }
27044     },
27045
27046     onItemClick : function(item, index, e)
27047     {
27048         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27049             return false;
27050         }
27051         if (this.toggleSelect) {
27052             var m = this.isSelected(item) ? 'unselect' : 'select';
27053             //Roo.log(m);
27054             var _t = this;
27055             _t[m](item, true, false);
27056             return true;
27057         }
27058         if(this.multiSelect || this.singleSelect){
27059             if(this.multiSelect && e.shiftKey && this.lastSelection){
27060                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27061             }else{
27062                 this.select(item, this.multiSelect && e.ctrlKey);
27063                 this.lastSelection = item;
27064             }
27065             
27066             if(!this.tickable){
27067                 e.preventDefault();
27068             }
27069             
27070         }
27071         return true;
27072     },
27073
27074     /**
27075      * Get the number of selected nodes.
27076      * @return {Number}
27077      */
27078     getSelectionCount : function(){
27079         return this.selections.length;
27080     },
27081
27082     /**
27083      * Get the currently selected nodes.
27084      * @return {Array} An array of HTMLElements
27085      */
27086     getSelectedNodes : function(){
27087         return this.selections;
27088     },
27089
27090     /**
27091      * Get the indexes of the selected nodes.
27092      * @return {Array}
27093      */
27094     getSelectedIndexes : function(){
27095         var indexes = [], s = this.selections;
27096         for(var i = 0, len = s.length; i < len; i++){
27097             indexes.push(s[i].nodeIndex);
27098         }
27099         return indexes;
27100     },
27101
27102     /**
27103      * Clear all selections
27104      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27105      */
27106     clearSelections : function(suppressEvent){
27107         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27108             this.cmp.elements = this.selections;
27109             this.cmp.removeClass(this.selectedClass);
27110             this.selections = [];
27111             if(!suppressEvent){
27112                 this.fireEvent("selectionchange", this, this.selections);
27113             }
27114         }
27115     },
27116
27117     /**
27118      * Returns true if the passed node is selected
27119      * @param {HTMLElement/Number} node The node or node index
27120      * @return {Boolean}
27121      */
27122     isSelected : function(node){
27123         var s = this.selections;
27124         if(s.length < 1){
27125             return false;
27126         }
27127         node = this.getNode(node);
27128         return s.indexOf(node) !== -1;
27129     },
27130
27131     /**
27132      * Selects nodes.
27133      * @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
27134      * @param {Boolean} keepExisting (optional) true to keep existing selections
27135      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27136      */
27137     select : function(nodeInfo, keepExisting, suppressEvent){
27138         if(nodeInfo instanceof Array){
27139             if(!keepExisting){
27140                 this.clearSelections(true);
27141             }
27142             for(var i = 0, len = nodeInfo.length; i < len; i++){
27143                 this.select(nodeInfo[i], true, true);
27144             }
27145             return;
27146         } 
27147         var node = this.getNode(nodeInfo);
27148         if(!node || this.isSelected(node)){
27149             return; // already selected.
27150         }
27151         if(!keepExisting){
27152             this.clearSelections(true);
27153         }
27154         
27155         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27156             Roo.fly(node).addClass(this.selectedClass);
27157             this.selections.push(node);
27158             if(!suppressEvent){
27159                 this.fireEvent("selectionchange", this, this.selections);
27160             }
27161         }
27162         
27163         
27164     },
27165       /**
27166      * Unselects nodes.
27167      * @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
27168      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27169      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27170      */
27171     unselect : function(nodeInfo, keepExisting, suppressEvent)
27172     {
27173         if(nodeInfo instanceof Array){
27174             Roo.each(this.selections, function(s) {
27175                 this.unselect(s, nodeInfo);
27176             }, this);
27177             return;
27178         }
27179         var node = this.getNode(nodeInfo);
27180         if(!node || !this.isSelected(node)){
27181             //Roo.log("not selected");
27182             return; // not selected.
27183         }
27184         // fireevent???
27185         var ns = [];
27186         Roo.each(this.selections, function(s) {
27187             if (s == node ) {
27188                 Roo.fly(node).removeClass(this.selectedClass);
27189
27190                 return;
27191             }
27192             ns.push(s);
27193         },this);
27194         
27195         this.selections= ns;
27196         this.fireEvent("selectionchange", this, this.selections);
27197     },
27198
27199     /**
27200      * Gets a template node.
27201      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27202      * @return {HTMLElement} The node or null if it wasn't found
27203      */
27204     getNode : function(nodeInfo){
27205         if(typeof nodeInfo == "string"){
27206             return document.getElementById(nodeInfo);
27207         }else if(typeof nodeInfo == "number"){
27208             return this.nodes[nodeInfo];
27209         }
27210         return nodeInfo;
27211     },
27212
27213     /**
27214      * Gets a range template nodes.
27215      * @param {Number} startIndex
27216      * @param {Number} endIndex
27217      * @return {Array} An array of nodes
27218      */
27219     getNodes : function(start, end){
27220         var ns = this.nodes;
27221         start = start || 0;
27222         end = typeof end == "undefined" ? ns.length - 1 : end;
27223         var nodes = [];
27224         if(start <= end){
27225             for(var i = start; i <= end; i++){
27226                 nodes.push(ns[i]);
27227             }
27228         } else{
27229             for(var i = start; i >= end; i--){
27230                 nodes.push(ns[i]);
27231             }
27232         }
27233         return nodes;
27234     },
27235
27236     /**
27237      * Finds the index of the passed node
27238      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27239      * @return {Number} The index of the node or -1
27240      */
27241     indexOf : function(node){
27242         node = this.getNode(node);
27243         if(typeof node.nodeIndex == "number"){
27244             return node.nodeIndex;
27245         }
27246         var ns = this.nodes;
27247         for(var i = 0, len = ns.length; i < len; i++){
27248             if(ns[i] == node){
27249                 return i;
27250             }
27251         }
27252         return -1;
27253     }
27254 });
27255 /*
27256  * Based on:
27257  * Ext JS Library 1.1.1
27258  * Copyright(c) 2006-2007, Ext JS, LLC.
27259  *
27260  * Originally Released Under LGPL - original licence link has changed is not relivant.
27261  *
27262  * Fork - LGPL
27263  * <script type="text/javascript">
27264  */
27265
27266 /**
27267  * @class Roo.JsonView
27268  * @extends Roo.View
27269  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27270 <pre><code>
27271 var view = new Roo.JsonView({
27272     container: "my-element",
27273     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27274     multiSelect: true, 
27275     jsonRoot: "data" 
27276 });
27277
27278 // listen for node click?
27279 view.on("click", function(vw, index, node, e){
27280     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27281 });
27282
27283 // direct load of JSON data
27284 view.load("foobar.php");
27285
27286 // Example from my blog list
27287 var tpl = new Roo.Template(
27288     '&lt;div class="entry"&gt;' +
27289     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27290     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27291     "&lt;/div&gt;&lt;hr /&gt;"
27292 );
27293
27294 var moreView = new Roo.JsonView({
27295     container :  "entry-list", 
27296     template : tpl,
27297     jsonRoot: "posts"
27298 });
27299 moreView.on("beforerender", this.sortEntries, this);
27300 moreView.load({
27301     url: "/blog/get-posts.php",
27302     params: "allposts=true",
27303     text: "Loading Blog Entries..."
27304 });
27305 </code></pre>
27306
27307 * Note: old code is supported with arguments : (container, template, config)
27308
27309
27310  * @constructor
27311  * Create a new JsonView
27312  * 
27313  * @param {Object} config The config object
27314  * 
27315  */
27316 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27317     
27318     
27319     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27320
27321     var um = this.el.getUpdateManager();
27322     um.setRenderer(this);
27323     um.on("update", this.onLoad, this);
27324     um.on("failure", this.onLoadException, this);
27325
27326     /**
27327      * @event beforerender
27328      * Fires before rendering of the downloaded JSON data.
27329      * @param {Roo.JsonView} this
27330      * @param {Object} data The JSON data loaded
27331      */
27332     /**
27333      * @event load
27334      * Fires when data is loaded.
27335      * @param {Roo.JsonView} this
27336      * @param {Object} data The JSON data loaded
27337      * @param {Object} response The raw Connect response object
27338      */
27339     /**
27340      * @event loadexception
27341      * Fires when loading fails.
27342      * @param {Roo.JsonView} this
27343      * @param {Object} response The raw Connect response object
27344      */
27345     this.addEvents({
27346         'beforerender' : true,
27347         'load' : true,
27348         'loadexception' : true
27349     });
27350 };
27351 Roo.extend(Roo.JsonView, Roo.View, {
27352     /**
27353      * @type {String} The root property in the loaded JSON object that contains the data
27354      */
27355     jsonRoot : "",
27356
27357     /**
27358      * Refreshes the view.
27359      */
27360     refresh : function(){
27361         this.clearSelections();
27362         this.el.update("");
27363         var html = [];
27364         var o = this.jsonData;
27365         if(o && o.length > 0){
27366             for(var i = 0, len = o.length; i < len; i++){
27367                 var data = this.prepareData(o[i], i, o);
27368                 html[html.length] = this.tpl.apply(data);
27369             }
27370         }else{
27371             html.push(this.emptyText);
27372         }
27373         this.el.update(html.join(""));
27374         this.nodes = this.el.dom.childNodes;
27375         this.updateIndexes(0);
27376     },
27377
27378     /**
27379      * 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.
27380      * @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:
27381      <pre><code>
27382      view.load({
27383          url: "your-url.php",
27384          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27385          callback: yourFunction,
27386          scope: yourObject, //(optional scope)
27387          discardUrl: false,
27388          nocache: false,
27389          text: "Loading...",
27390          timeout: 30,
27391          scripts: false
27392      });
27393      </code></pre>
27394      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27395      * 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.
27396      * @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}
27397      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27398      * @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.
27399      */
27400     load : function(){
27401         var um = this.el.getUpdateManager();
27402         um.update.apply(um, arguments);
27403     },
27404
27405     // note - render is a standard framework call...
27406     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27407     render : function(el, response){
27408         
27409         this.clearSelections();
27410         this.el.update("");
27411         var o;
27412         try{
27413             if (response != '') {
27414                 o = Roo.util.JSON.decode(response.responseText);
27415                 if(this.jsonRoot){
27416                     
27417                     o = o[this.jsonRoot];
27418                 }
27419             }
27420         } catch(e){
27421         }
27422         /**
27423          * The current JSON data or null
27424          */
27425         this.jsonData = o;
27426         this.beforeRender();
27427         this.refresh();
27428     },
27429
27430 /**
27431  * Get the number of records in the current JSON dataset
27432  * @return {Number}
27433  */
27434     getCount : function(){
27435         return this.jsonData ? this.jsonData.length : 0;
27436     },
27437
27438 /**
27439  * Returns the JSON object for the specified node(s)
27440  * @param {HTMLElement/Array} node The node or an array of nodes
27441  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27442  * you get the JSON object for the node
27443  */
27444     getNodeData : function(node){
27445         if(node instanceof Array){
27446             var data = [];
27447             for(var i = 0, len = node.length; i < len; i++){
27448                 data.push(this.getNodeData(node[i]));
27449             }
27450             return data;
27451         }
27452         return this.jsonData[this.indexOf(node)] || null;
27453     },
27454
27455     beforeRender : function(){
27456         this.snapshot = this.jsonData;
27457         if(this.sortInfo){
27458             this.sort.apply(this, this.sortInfo);
27459         }
27460         this.fireEvent("beforerender", this, this.jsonData);
27461     },
27462
27463     onLoad : function(el, o){
27464         this.fireEvent("load", this, this.jsonData, o);
27465     },
27466
27467     onLoadException : function(el, o){
27468         this.fireEvent("loadexception", this, o);
27469     },
27470
27471 /**
27472  * Filter the data by a specific property.
27473  * @param {String} property A property on your JSON objects
27474  * @param {String/RegExp} value Either string that the property values
27475  * should start with, or a RegExp to test against the property
27476  */
27477     filter : function(property, value){
27478         if(this.jsonData){
27479             var data = [];
27480             var ss = this.snapshot;
27481             if(typeof value == "string"){
27482                 var vlen = value.length;
27483                 if(vlen == 0){
27484                     this.clearFilter();
27485                     return;
27486                 }
27487                 value = value.toLowerCase();
27488                 for(var i = 0, len = ss.length; i < len; i++){
27489                     var o = ss[i];
27490                     if(o[property].substr(0, vlen).toLowerCase() == value){
27491                         data.push(o);
27492                     }
27493                 }
27494             } else if(value.exec){ // regex?
27495                 for(var i = 0, len = ss.length; i < len; i++){
27496                     var o = ss[i];
27497                     if(value.test(o[property])){
27498                         data.push(o);
27499                     }
27500                 }
27501             } else{
27502                 return;
27503             }
27504             this.jsonData = data;
27505             this.refresh();
27506         }
27507     },
27508
27509 /**
27510  * Filter by a function. The passed function will be called with each
27511  * object in the current dataset. If the function returns true the value is kept,
27512  * otherwise it is filtered.
27513  * @param {Function} fn
27514  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27515  */
27516     filterBy : function(fn, scope){
27517         if(this.jsonData){
27518             var data = [];
27519             var ss = this.snapshot;
27520             for(var i = 0, len = ss.length; i < len; i++){
27521                 var o = ss[i];
27522                 if(fn.call(scope || this, o)){
27523                     data.push(o);
27524                 }
27525             }
27526             this.jsonData = data;
27527             this.refresh();
27528         }
27529     },
27530
27531 /**
27532  * Clears the current filter.
27533  */
27534     clearFilter : function(){
27535         if(this.snapshot && this.jsonData != this.snapshot){
27536             this.jsonData = this.snapshot;
27537             this.refresh();
27538         }
27539     },
27540
27541
27542 /**
27543  * Sorts the data for this view and refreshes it.
27544  * @param {String} property A property on your JSON objects to sort on
27545  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27546  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27547  */
27548     sort : function(property, dir, sortType){
27549         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27550         if(this.jsonData){
27551             var p = property;
27552             var dsc = dir && dir.toLowerCase() == "desc";
27553             var f = function(o1, o2){
27554                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27555                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27556                 ;
27557                 if(v1 < v2){
27558                     return dsc ? +1 : -1;
27559                 } else if(v1 > v2){
27560                     return dsc ? -1 : +1;
27561                 } else{
27562                     return 0;
27563                 }
27564             };
27565             this.jsonData.sort(f);
27566             this.refresh();
27567             if(this.jsonData != this.snapshot){
27568                 this.snapshot.sort(f);
27569             }
27570         }
27571     }
27572 });/*
27573  * Based on:
27574  * Ext JS Library 1.1.1
27575  * Copyright(c) 2006-2007, Ext JS, LLC.
27576  *
27577  * Originally Released Under LGPL - original licence link has changed is not relivant.
27578  *
27579  * Fork - LGPL
27580  * <script type="text/javascript">
27581  */
27582  
27583
27584 /**
27585  * @class Roo.ColorPalette
27586  * @extends Roo.Component
27587  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27588  * Here's an example of typical usage:
27589  * <pre><code>
27590 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27591 cp.render('my-div');
27592
27593 cp.on('select', function(palette, selColor){
27594     // do something with selColor
27595 });
27596 </code></pre>
27597  * @constructor
27598  * Create a new ColorPalette
27599  * @param {Object} config The config object
27600  */
27601 Roo.ColorPalette = function(config){
27602     Roo.ColorPalette.superclass.constructor.call(this, config);
27603     this.addEvents({
27604         /**
27605              * @event select
27606              * Fires when a color is selected
27607              * @param {ColorPalette} this
27608              * @param {String} color The 6-digit color hex code (without the # symbol)
27609              */
27610         select: true
27611     });
27612
27613     if(this.handler){
27614         this.on("select", this.handler, this.scope, true);
27615     }
27616 };
27617 Roo.extend(Roo.ColorPalette, Roo.Component, {
27618     /**
27619      * @cfg {String} itemCls
27620      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27621      */
27622     itemCls : "x-color-palette",
27623     /**
27624      * @cfg {String} value
27625      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27626      * the hex codes are case-sensitive.
27627      */
27628     value : null,
27629     clickEvent:'click',
27630     // private
27631     ctype: "Roo.ColorPalette",
27632
27633     /**
27634      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27635      */
27636     allowReselect : false,
27637
27638     /**
27639      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27640      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27641      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27642      * of colors with the width setting until the box is symmetrical.</p>
27643      * <p>You can override individual colors if needed:</p>
27644      * <pre><code>
27645 var cp = new Roo.ColorPalette();
27646 cp.colors[0] = "FF0000";  // change the first box to red
27647 </code></pre>
27648
27649 Or you can provide a custom array of your own for complete control:
27650 <pre><code>
27651 var cp = new Roo.ColorPalette();
27652 cp.colors = ["000000", "993300", "333300"];
27653 </code></pre>
27654      * @type Array
27655      */
27656     colors : [
27657         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27658         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27659         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27660         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27661         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27662     ],
27663
27664     // private
27665     onRender : function(container, position){
27666         var t = new Roo.MasterTemplate(
27667             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27668         );
27669         var c = this.colors;
27670         for(var i = 0, len = c.length; i < len; i++){
27671             t.add([c[i]]);
27672         }
27673         var el = document.createElement("div");
27674         el.className = this.itemCls;
27675         t.overwrite(el);
27676         container.dom.insertBefore(el, position);
27677         this.el = Roo.get(el);
27678         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27679         if(this.clickEvent != 'click'){
27680             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27681         }
27682     },
27683
27684     // private
27685     afterRender : function(){
27686         Roo.ColorPalette.superclass.afterRender.call(this);
27687         if(this.value){
27688             var s = this.value;
27689             this.value = null;
27690             this.select(s);
27691         }
27692     },
27693
27694     // private
27695     handleClick : function(e, t){
27696         e.preventDefault();
27697         if(!this.disabled){
27698             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27699             this.select(c.toUpperCase());
27700         }
27701     },
27702
27703     /**
27704      * Selects the specified color in the palette (fires the select event)
27705      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27706      */
27707     select : function(color){
27708         color = color.replace("#", "");
27709         if(color != this.value || this.allowReselect){
27710             var el = this.el;
27711             if(this.value){
27712                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27713             }
27714             el.child("a.color-"+color).addClass("x-color-palette-sel");
27715             this.value = color;
27716             this.fireEvent("select", this, color);
27717         }
27718     }
27719 });/*
27720  * Based on:
27721  * Ext JS Library 1.1.1
27722  * Copyright(c) 2006-2007, Ext JS, LLC.
27723  *
27724  * Originally Released Under LGPL - original licence link has changed is not relivant.
27725  *
27726  * Fork - LGPL
27727  * <script type="text/javascript">
27728  */
27729  
27730 /**
27731  * @class Roo.DatePicker
27732  * @extends Roo.Component
27733  * Simple date picker class.
27734  * @constructor
27735  * Create a new DatePicker
27736  * @param {Object} config The config object
27737  */
27738 Roo.DatePicker = function(config){
27739     Roo.DatePicker.superclass.constructor.call(this, config);
27740
27741     this.value = config && config.value ?
27742                  config.value.clearTime() : new Date().clearTime();
27743
27744     this.addEvents({
27745         /**
27746              * @event select
27747              * Fires when a date is selected
27748              * @param {DatePicker} this
27749              * @param {Date} date The selected date
27750              */
27751         'select': true,
27752         /**
27753              * @event monthchange
27754              * Fires when the displayed month changes 
27755              * @param {DatePicker} this
27756              * @param {Date} date The selected month
27757              */
27758         'monthchange': true
27759     });
27760
27761     if(this.handler){
27762         this.on("select", this.handler,  this.scope || this);
27763     }
27764     // build the disabledDatesRE
27765     if(!this.disabledDatesRE && this.disabledDates){
27766         var dd = this.disabledDates;
27767         var re = "(?:";
27768         for(var i = 0; i < dd.length; i++){
27769             re += dd[i];
27770             if(i != dd.length-1) {
27771                 re += "|";
27772             }
27773         }
27774         this.disabledDatesRE = new RegExp(re + ")");
27775     }
27776 };
27777
27778 Roo.extend(Roo.DatePicker, Roo.Component, {
27779     /**
27780      * @cfg {String} todayText
27781      * The text to display on the button that selects the current date (defaults to "Today")
27782      */
27783     todayText : "Today",
27784     /**
27785      * @cfg {String} okText
27786      * The text to display on the ok button
27787      */
27788     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27789     /**
27790      * @cfg {String} cancelText
27791      * The text to display on the cancel button
27792      */
27793     cancelText : "Cancel",
27794     /**
27795      * @cfg {String} todayTip
27796      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27797      */
27798     todayTip : "{0} (Spacebar)",
27799     /**
27800      * @cfg {Date} minDate
27801      * Minimum allowable date (JavaScript date object, defaults to null)
27802      */
27803     minDate : null,
27804     /**
27805      * @cfg {Date} maxDate
27806      * Maximum allowable date (JavaScript date object, defaults to null)
27807      */
27808     maxDate : null,
27809     /**
27810      * @cfg {String} minText
27811      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27812      */
27813     minText : "This date is before the minimum date",
27814     /**
27815      * @cfg {String} maxText
27816      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27817      */
27818     maxText : "This date is after the maximum date",
27819     /**
27820      * @cfg {String} format
27821      * The default date format string which can be overriden for localization support.  The format must be
27822      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27823      */
27824     format : "m/d/y",
27825     /**
27826      * @cfg {Array} disabledDays
27827      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27828      */
27829     disabledDays : null,
27830     /**
27831      * @cfg {String} disabledDaysText
27832      * The tooltip to display when the date falls on a disabled day (defaults to "")
27833      */
27834     disabledDaysText : "",
27835     /**
27836      * @cfg {RegExp} disabledDatesRE
27837      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27838      */
27839     disabledDatesRE : null,
27840     /**
27841      * @cfg {String} disabledDatesText
27842      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27843      */
27844     disabledDatesText : "",
27845     /**
27846      * @cfg {Boolean} constrainToViewport
27847      * True to constrain the date picker to the viewport (defaults to true)
27848      */
27849     constrainToViewport : true,
27850     /**
27851      * @cfg {Array} monthNames
27852      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27853      */
27854     monthNames : Date.monthNames,
27855     /**
27856      * @cfg {Array} dayNames
27857      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27858      */
27859     dayNames : Date.dayNames,
27860     /**
27861      * @cfg {String} nextText
27862      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27863      */
27864     nextText: 'Next Month (Control+Right)',
27865     /**
27866      * @cfg {String} prevText
27867      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27868      */
27869     prevText: 'Previous Month (Control+Left)',
27870     /**
27871      * @cfg {String} monthYearText
27872      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27873      */
27874     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27875     /**
27876      * @cfg {Number} startDay
27877      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27878      */
27879     startDay : 0,
27880     /**
27881      * @cfg {Bool} showClear
27882      * Show a clear button (usefull for date form elements that can be blank.)
27883      */
27884     
27885     showClear: false,
27886     
27887     /**
27888      * Sets the value of the date field
27889      * @param {Date} value The date to set
27890      */
27891     setValue : function(value){
27892         var old = this.value;
27893         
27894         if (typeof(value) == 'string') {
27895          
27896             value = Date.parseDate(value, this.format);
27897         }
27898         if (!value) {
27899             value = new Date();
27900         }
27901         
27902         this.value = value.clearTime(true);
27903         if(this.el){
27904             this.update(this.value);
27905         }
27906     },
27907
27908     /**
27909      * Gets the current selected value of the date field
27910      * @return {Date} The selected date
27911      */
27912     getValue : function(){
27913         return this.value;
27914     },
27915
27916     // private
27917     focus : function(){
27918         if(this.el){
27919             this.update(this.activeDate);
27920         }
27921     },
27922
27923     // privateval
27924     onRender : function(container, position){
27925         
27926         var m = [
27927              '<table cellspacing="0">',
27928                 '<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>',
27929                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27930         var dn = this.dayNames;
27931         for(var i = 0; i < 7; i++){
27932             var d = this.startDay+i;
27933             if(d > 6){
27934                 d = d-7;
27935             }
27936             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27937         }
27938         m[m.length] = "</tr></thead><tbody><tr>";
27939         for(var i = 0; i < 42; i++) {
27940             if(i % 7 == 0 && i != 0){
27941                 m[m.length] = "</tr><tr>";
27942             }
27943             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27944         }
27945         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27946             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27947
27948         var el = document.createElement("div");
27949         el.className = "x-date-picker";
27950         el.innerHTML = m.join("");
27951
27952         container.dom.insertBefore(el, position);
27953
27954         this.el = Roo.get(el);
27955         this.eventEl = Roo.get(el.firstChild);
27956
27957         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27958             handler: this.showPrevMonth,
27959             scope: this,
27960             preventDefault:true,
27961             stopDefault:true
27962         });
27963
27964         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27965             handler: this.showNextMonth,
27966             scope: this,
27967             preventDefault:true,
27968             stopDefault:true
27969         });
27970
27971         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27972
27973         this.monthPicker = this.el.down('div.x-date-mp');
27974         this.monthPicker.enableDisplayMode('block');
27975         
27976         var kn = new Roo.KeyNav(this.eventEl, {
27977             "left" : function(e){
27978                 e.ctrlKey ?
27979                     this.showPrevMonth() :
27980                     this.update(this.activeDate.add("d", -1));
27981             },
27982
27983             "right" : function(e){
27984                 e.ctrlKey ?
27985                     this.showNextMonth() :
27986                     this.update(this.activeDate.add("d", 1));
27987             },
27988
27989             "up" : function(e){
27990                 e.ctrlKey ?
27991                     this.showNextYear() :
27992                     this.update(this.activeDate.add("d", -7));
27993             },
27994
27995             "down" : function(e){
27996                 e.ctrlKey ?
27997                     this.showPrevYear() :
27998                     this.update(this.activeDate.add("d", 7));
27999             },
28000
28001             "pageUp" : function(e){
28002                 this.showNextMonth();
28003             },
28004
28005             "pageDown" : function(e){
28006                 this.showPrevMonth();
28007             },
28008
28009             "enter" : function(e){
28010                 e.stopPropagation();
28011                 return true;
28012             },
28013
28014             scope : this
28015         });
28016
28017         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
28018
28019         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
28020
28021         this.el.unselectable();
28022         
28023         this.cells = this.el.select("table.x-date-inner tbody td");
28024         this.textNodes = this.el.query("table.x-date-inner tbody span");
28025
28026         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
28027             text: "&#160;",
28028             tooltip: this.monthYearText
28029         });
28030
28031         this.mbtn.on('click', this.showMonthPicker, this);
28032         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
28033
28034
28035         var today = (new Date()).dateFormat(this.format);
28036         
28037         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
28038         if (this.showClear) {
28039             baseTb.add( new Roo.Toolbar.Fill());
28040         }
28041         baseTb.add({
28042             text: String.format(this.todayText, today),
28043             tooltip: String.format(this.todayTip, today),
28044             handler: this.selectToday,
28045             scope: this
28046         });
28047         
28048         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28049             
28050         //});
28051         if (this.showClear) {
28052             
28053             baseTb.add( new Roo.Toolbar.Fill());
28054             baseTb.add({
28055                 text: '&#160;',
28056                 cls: 'x-btn-icon x-btn-clear',
28057                 handler: function() {
28058                     //this.value = '';
28059                     this.fireEvent("select", this, '');
28060                 },
28061                 scope: this
28062             });
28063         }
28064         
28065         
28066         if(Roo.isIE){
28067             this.el.repaint();
28068         }
28069         this.update(this.value);
28070     },
28071
28072     createMonthPicker : function(){
28073         if(!this.monthPicker.dom.firstChild){
28074             var buf = ['<table border="0" cellspacing="0">'];
28075             for(var i = 0; i < 6; i++){
28076                 buf.push(
28077                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28078                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28079                     i == 0 ?
28080                     '<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>' :
28081                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28082                 );
28083             }
28084             buf.push(
28085                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28086                     this.okText,
28087                     '</button><button type="button" class="x-date-mp-cancel">',
28088                     this.cancelText,
28089                     '</button></td></tr>',
28090                 '</table>'
28091             );
28092             this.monthPicker.update(buf.join(''));
28093             this.monthPicker.on('click', this.onMonthClick, this);
28094             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28095
28096             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28097             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28098
28099             this.mpMonths.each(function(m, a, i){
28100                 i += 1;
28101                 if((i%2) == 0){
28102                     m.dom.xmonth = 5 + Math.round(i * .5);
28103                 }else{
28104                     m.dom.xmonth = Math.round((i-1) * .5);
28105                 }
28106             });
28107         }
28108     },
28109
28110     showMonthPicker : function(){
28111         this.createMonthPicker();
28112         var size = this.el.getSize();
28113         this.monthPicker.setSize(size);
28114         this.monthPicker.child('table').setSize(size);
28115
28116         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28117         this.updateMPMonth(this.mpSelMonth);
28118         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28119         this.updateMPYear(this.mpSelYear);
28120
28121         this.monthPicker.slideIn('t', {duration:.2});
28122     },
28123
28124     updateMPYear : function(y){
28125         this.mpyear = y;
28126         var ys = this.mpYears.elements;
28127         for(var i = 1; i <= 10; i++){
28128             var td = ys[i-1], y2;
28129             if((i%2) == 0){
28130                 y2 = y + Math.round(i * .5);
28131                 td.firstChild.innerHTML = y2;
28132                 td.xyear = y2;
28133             }else{
28134                 y2 = y - (5-Math.round(i * .5));
28135                 td.firstChild.innerHTML = y2;
28136                 td.xyear = y2;
28137             }
28138             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28139         }
28140     },
28141
28142     updateMPMonth : function(sm){
28143         this.mpMonths.each(function(m, a, i){
28144             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28145         });
28146     },
28147
28148     selectMPMonth: function(m){
28149         
28150     },
28151
28152     onMonthClick : function(e, t){
28153         e.stopEvent();
28154         var el = new Roo.Element(t), pn;
28155         if(el.is('button.x-date-mp-cancel')){
28156             this.hideMonthPicker();
28157         }
28158         else if(el.is('button.x-date-mp-ok')){
28159             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28160             this.hideMonthPicker();
28161         }
28162         else if(pn = el.up('td.x-date-mp-month', 2)){
28163             this.mpMonths.removeClass('x-date-mp-sel');
28164             pn.addClass('x-date-mp-sel');
28165             this.mpSelMonth = pn.dom.xmonth;
28166         }
28167         else if(pn = el.up('td.x-date-mp-year', 2)){
28168             this.mpYears.removeClass('x-date-mp-sel');
28169             pn.addClass('x-date-mp-sel');
28170             this.mpSelYear = pn.dom.xyear;
28171         }
28172         else if(el.is('a.x-date-mp-prev')){
28173             this.updateMPYear(this.mpyear-10);
28174         }
28175         else if(el.is('a.x-date-mp-next')){
28176             this.updateMPYear(this.mpyear+10);
28177         }
28178     },
28179
28180     onMonthDblClick : function(e, t){
28181         e.stopEvent();
28182         var el = new Roo.Element(t), pn;
28183         if(pn = el.up('td.x-date-mp-month', 2)){
28184             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28185             this.hideMonthPicker();
28186         }
28187         else if(pn = el.up('td.x-date-mp-year', 2)){
28188             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28189             this.hideMonthPicker();
28190         }
28191     },
28192
28193     hideMonthPicker : function(disableAnim){
28194         if(this.monthPicker){
28195             if(disableAnim === true){
28196                 this.monthPicker.hide();
28197             }else{
28198                 this.monthPicker.slideOut('t', {duration:.2});
28199             }
28200         }
28201     },
28202
28203     // private
28204     showPrevMonth : function(e){
28205         this.update(this.activeDate.add("mo", -1));
28206     },
28207
28208     // private
28209     showNextMonth : function(e){
28210         this.update(this.activeDate.add("mo", 1));
28211     },
28212
28213     // private
28214     showPrevYear : function(){
28215         this.update(this.activeDate.add("y", -1));
28216     },
28217
28218     // private
28219     showNextYear : function(){
28220         this.update(this.activeDate.add("y", 1));
28221     },
28222
28223     // private
28224     handleMouseWheel : function(e){
28225         var delta = e.getWheelDelta();
28226         if(delta > 0){
28227             this.showPrevMonth();
28228             e.stopEvent();
28229         } else if(delta < 0){
28230             this.showNextMonth();
28231             e.stopEvent();
28232         }
28233     },
28234
28235     // private
28236     handleDateClick : function(e, t){
28237         e.stopEvent();
28238         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28239             this.setValue(new Date(t.dateValue));
28240             this.fireEvent("select", this, this.value);
28241         }
28242     },
28243
28244     // private
28245     selectToday : function(){
28246         this.setValue(new Date().clearTime());
28247         this.fireEvent("select", this, this.value);
28248     },
28249
28250     // private
28251     update : function(date)
28252     {
28253         var vd = this.activeDate;
28254         this.activeDate = date;
28255         if(vd && this.el){
28256             var t = date.getTime();
28257             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28258                 this.cells.removeClass("x-date-selected");
28259                 this.cells.each(function(c){
28260                    if(c.dom.firstChild.dateValue == t){
28261                        c.addClass("x-date-selected");
28262                        setTimeout(function(){
28263                             try{c.dom.firstChild.focus();}catch(e){}
28264                        }, 50);
28265                        return false;
28266                    }
28267                 });
28268                 return;
28269             }
28270         }
28271         
28272         var days = date.getDaysInMonth();
28273         var firstOfMonth = date.getFirstDateOfMonth();
28274         var startingPos = firstOfMonth.getDay()-this.startDay;
28275
28276         if(startingPos <= this.startDay){
28277             startingPos += 7;
28278         }
28279
28280         var pm = date.add("mo", -1);
28281         var prevStart = pm.getDaysInMonth()-startingPos;
28282
28283         var cells = this.cells.elements;
28284         var textEls = this.textNodes;
28285         days += startingPos;
28286
28287         // convert everything to numbers so it's fast
28288         var day = 86400000;
28289         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28290         var today = new Date().clearTime().getTime();
28291         var sel = date.clearTime().getTime();
28292         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28293         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28294         var ddMatch = this.disabledDatesRE;
28295         var ddText = this.disabledDatesText;
28296         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28297         var ddaysText = this.disabledDaysText;
28298         var format = this.format;
28299
28300         var setCellClass = function(cal, cell){
28301             cell.title = "";
28302             var t = d.getTime();
28303             cell.firstChild.dateValue = t;
28304             if(t == today){
28305                 cell.className += " x-date-today";
28306                 cell.title = cal.todayText;
28307             }
28308             if(t == sel){
28309                 cell.className += " x-date-selected";
28310                 setTimeout(function(){
28311                     try{cell.firstChild.focus();}catch(e){}
28312                 }, 50);
28313             }
28314             // disabling
28315             if(t < min) {
28316                 cell.className = " x-date-disabled";
28317                 cell.title = cal.minText;
28318                 return;
28319             }
28320             if(t > max) {
28321                 cell.className = " x-date-disabled";
28322                 cell.title = cal.maxText;
28323                 return;
28324             }
28325             if(ddays){
28326                 if(ddays.indexOf(d.getDay()) != -1){
28327                     cell.title = ddaysText;
28328                     cell.className = " x-date-disabled";
28329                 }
28330             }
28331             if(ddMatch && format){
28332                 var fvalue = d.dateFormat(format);
28333                 if(ddMatch.test(fvalue)){
28334                     cell.title = ddText.replace("%0", fvalue);
28335                     cell.className = " x-date-disabled";
28336                 }
28337             }
28338         };
28339
28340         var i = 0;
28341         for(; i < startingPos; i++) {
28342             textEls[i].innerHTML = (++prevStart);
28343             d.setDate(d.getDate()+1);
28344             cells[i].className = "x-date-prevday";
28345             setCellClass(this, cells[i]);
28346         }
28347         for(; i < days; i++){
28348             intDay = i - startingPos + 1;
28349             textEls[i].innerHTML = (intDay);
28350             d.setDate(d.getDate()+1);
28351             cells[i].className = "x-date-active";
28352             setCellClass(this, cells[i]);
28353         }
28354         var extraDays = 0;
28355         for(; i < 42; i++) {
28356              textEls[i].innerHTML = (++extraDays);
28357              d.setDate(d.getDate()+1);
28358              cells[i].className = "x-date-nextday";
28359              setCellClass(this, cells[i]);
28360         }
28361
28362         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28363         this.fireEvent('monthchange', this, date);
28364         
28365         if(!this.internalRender){
28366             var main = this.el.dom.firstChild;
28367             var w = main.offsetWidth;
28368             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28369             Roo.fly(main).setWidth(w);
28370             this.internalRender = true;
28371             // opera does not respect the auto grow header center column
28372             // then, after it gets a width opera refuses to recalculate
28373             // without a second pass
28374             if(Roo.isOpera && !this.secondPass){
28375                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28376                 this.secondPass = true;
28377                 this.update.defer(10, this, [date]);
28378             }
28379         }
28380         
28381         
28382     }
28383 });        /*
28384  * Based on:
28385  * Ext JS Library 1.1.1
28386  * Copyright(c) 2006-2007, Ext JS, LLC.
28387  *
28388  * Originally Released Under LGPL - original licence link has changed is not relivant.
28389  *
28390  * Fork - LGPL
28391  * <script type="text/javascript">
28392  */
28393 /**
28394  * @class Roo.TabPanel
28395  * @extends Roo.util.Observable
28396  * A lightweight tab container.
28397  * <br><br>
28398  * Usage:
28399  * <pre><code>
28400 // basic tabs 1, built from existing content
28401 var tabs = new Roo.TabPanel("tabs1");
28402 tabs.addTab("script", "View Script");
28403 tabs.addTab("markup", "View Markup");
28404 tabs.activate("script");
28405
28406 // more advanced tabs, built from javascript
28407 var jtabs = new Roo.TabPanel("jtabs");
28408 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28409
28410 // set up the UpdateManager
28411 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28412 var updater = tab2.getUpdateManager();
28413 updater.setDefaultUrl("ajax1.htm");
28414 tab2.on('activate', updater.refresh, updater, true);
28415
28416 // Use setUrl for Ajax loading
28417 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28418 tab3.setUrl("ajax2.htm", null, true);
28419
28420 // Disabled tab
28421 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28422 tab4.disable();
28423
28424 jtabs.activate("jtabs-1");
28425  * </code></pre>
28426  * @constructor
28427  * Create a new TabPanel.
28428  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28429  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28430  */
28431 Roo.TabPanel = function(container, config){
28432     /**
28433     * The container element for this TabPanel.
28434     * @type Roo.Element
28435     */
28436     this.el = Roo.get(container, true);
28437     if(config){
28438         if(typeof config == "boolean"){
28439             this.tabPosition = config ? "bottom" : "top";
28440         }else{
28441             Roo.apply(this, config);
28442         }
28443     }
28444     if(this.tabPosition == "bottom"){
28445         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28446         this.el.addClass("x-tabs-bottom");
28447     }
28448     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28449     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28450     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28451     if(Roo.isIE){
28452         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28453     }
28454     if(this.tabPosition != "bottom"){
28455         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28456          * @type Roo.Element
28457          */
28458         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28459         this.el.addClass("x-tabs-top");
28460     }
28461     this.items = [];
28462
28463     this.bodyEl.setStyle("position", "relative");
28464
28465     this.active = null;
28466     this.activateDelegate = this.activate.createDelegate(this);
28467
28468     this.addEvents({
28469         /**
28470          * @event tabchange
28471          * Fires when the active tab changes
28472          * @param {Roo.TabPanel} this
28473          * @param {Roo.TabPanelItem} activePanel The new active tab
28474          */
28475         "tabchange": true,
28476         /**
28477          * @event beforetabchange
28478          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28479          * @param {Roo.TabPanel} this
28480          * @param {Object} e Set cancel to true on this object to cancel the tab change
28481          * @param {Roo.TabPanelItem} tab The tab being changed to
28482          */
28483         "beforetabchange" : true
28484     });
28485
28486     Roo.EventManager.onWindowResize(this.onResize, this);
28487     this.cpad = this.el.getPadding("lr");
28488     this.hiddenCount = 0;
28489
28490
28491     // toolbar on the tabbar support...
28492     if (this.toolbar) {
28493         var tcfg = this.toolbar;
28494         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28495         this.toolbar = new Roo.Toolbar(tcfg);
28496         if (Roo.isSafari) {
28497             var tbl = tcfg.container.child('table', true);
28498             tbl.setAttribute('width', '100%');
28499         }
28500         
28501     }
28502    
28503
28504
28505     Roo.TabPanel.superclass.constructor.call(this);
28506 };
28507
28508 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28509     /*
28510      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28511      */
28512     tabPosition : "top",
28513     /*
28514      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28515      */
28516     currentTabWidth : 0,
28517     /*
28518      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28519      */
28520     minTabWidth : 40,
28521     /*
28522      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28523      */
28524     maxTabWidth : 250,
28525     /*
28526      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28527      */
28528     preferredTabWidth : 175,
28529     /*
28530      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28531      */
28532     resizeTabs : false,
28533     /*
28534      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28535      */
28536     monitorResize : true,
28537     /*
28538      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28539      */
28540     toolbar : false,
28541
28542     /**
28543      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28544      * @param {String} id The id of the div to use <b>or create</b>
28545      * @param {String} text The text for the tab
28546      * @param {String} content (optional) Content to put in the TabPanelItem body
28547      * @param {Boolean} closable (optional) True to create a close icon on the tab
28548      * @return {Roo.TabPanelItem} The created TabPanelItem
28549      */
28550     addTab : function(id, text, content, closable){
28551         var item = new Roo.TabPanelItem(this, id, text, closable);
28552         this.addTabItem(item);
28553         if(content){
28554             item.setContent(content);
28555         }
28556         return item;
28557     },
28558
28559     /**
28560      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28561      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28562      * @return {Roo.TabPanelItem}
28563      */
28564     getTab : function(id){
28565         return this.items[id];
28566     },
28567
28568     /**
28569      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28570      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28571      */
28572     hideTab : function(id){
28573         var t = this.items[id];
28574         if(!t.isHidden()){
28575            t.setHidden(true);
28576            this.hiddenCount++;
28577            this.autoSizeTabs();
28578         }
28579     },
28580
28581     /**
28582      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28583      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28584      */
28585     unhideTab : function(id){
28586         var t = this.items[id];
28587         if(t.isHidden()){
28588            t.setHidden(false);
28589            this.hiddenCount--;
28590            this.autoSizeTabs();
28591         }
28592     },
28593
28594     /**
28595      * Adds an existing {@link Roo.TabPanelItem}.
28596      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28597      */
28598     addTabItem : function(item){
28599         this.items[item.id] = item;
28600         this.items.push(item);
28601         if(this.resizeTabs){
28602            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28603            this.autoSizeTabs();
28604         }else{
28605             item.autoSize();
28606         }
28607     },
28608
28609     /**
28610      * Removes a {@link Roo.TabPanelItem}.
28611      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28612      */
28613     removeTab : function(id){
28614         var items = this.items;
28615         var tab = items[id];
28616         if(!tab) { return; }
28617         var index = items.indexOf(tab);
28618         if(this.active == tab && items.length > 1){
28619             var newTab = this.getNextAvailable(index);
28620             if(newTab) {
28621                 newTab.activate();
28622             }
28623         }
28624         this.stripEl.dom.removeChild(tab.pnode.dom);
28625         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28626             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28627         }
28628         items.splice(index, 1);
28629         delete this.items[tab.id];
28630         tab.fireEvent("close", tab);
28631         tab.purgeListeners();
28632         this.autoSizeTabs();
28633     },
28634
28635     getNextAvailable : function(start){
28636         var items = this.items;
28637         var index = start;
28638         // look for a next tab that will slide over to
28639         // replace the one being removed
28640         while(index < items.length){
28641             var item = items[++index];
28642             if(item && !item.isHidden()){
28643                 return item;
28644             }
28645         }
28646         // if one isn't found select the previous tab (on the left)
28647         index = start;
28648         while(index >= 0){
28649             var item = items[--index];
28650             if(item && !item.isHidden()){
28651                 return item;
28652             }
28653         }
28654         return null;
28655     },
28656
28657     /**
28658      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28659      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28660      */
28661     disableTab : function(id){
28662         var tab = this.items[id];
28663         if(tab && this.active != tab){
28664             tab.disable();
28665         }
28666     },
28667
28668     /**
28669      * Enables a {@link Roo.TabPanelItem} that is disabled.
28670      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28671      */
28672     enableTab : function(id){
28673         var tab = this.items[id];
28674         tab.enable();
28675     },
28676
28677     /**
28678      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28679      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28680      * @return {Roo.TabPanelItem} The TabPanelItem.
28681      */
28682     activate : function(id){
28683         var tab = this.items[id];
28684         if(!tab){
28685             return null;
28686         }
28687         if(tab == this.active || tab.disabled){
28688             return tab;
28689         }
28690         var e = {};
28691         this.fireEvent("beforetabchange", this, e, tab);
28692         if(e.cancel !== true && !tab.disabled){
28693             if(this.active){
28694                 this.active.hide();
28695             }
28696             this.active = this.items[id];
28697             this.active.show();
28698             this.fireEvent("tabchange", this, this.active);
28699         }
28700         return tab;
28701     },
28702
28703     /**
28704      * Gets the active {@link Roo.TabPanelItem}.
28705      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28706      */
28707     getActiveTab : function(){
28708         return this.active;
28709     },
28710
28711     /**
28712      * Updates the tab body element to fit the height of the container element
28713      * for overflow scrolling
28714      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28715      */
28716     syncHeight : function(targetHeight){
28717         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28718         var bm = this.bodyEl.getMargins();
28719         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28720         this.bodyEl.setHeight(newHeight);
28721         return newHeight;
28722     },
28723
28724     onResize : function(){
28725         if(this.monitorResize){
28726             this.autoSizeTabs();
28727         }
28728     },
28729
28730     /**
28731      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28732      */
28733     beginUpdate : function(){
28734         this.updating = true;
28735     },
28736
28737     /**
28738      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28739      */
28740     endUpdate : function(){
28741         this.updating = false;
28742         this.autoSizeTabs();
28743     },
28744
28745     /**
28746      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28747      */
28748     autoSizeTabs : function(){
28749         var count = this.items.length;
28750         var vcount = count - this.hiddenCount;
28751         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28752             return;
28753         }
28754         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28755         var availWidth = Math.floor(w / vcount);
28756         var b = this.stripBody;
28757         if(b.getWidth() > w){
28758             var tabs = this.items;
28759             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28760             if(availWidth < this.minTabWidth){
28761                 /*if(!this.sleft){    // incomplete scrolling code
28762                     this.createScrollButtons();
28763                 }
28764                 this.showScroll();
28765                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28766             }
28767         }else{
28768             if(this.currentTabWidth < this.preferredTabWidth){
28769                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28770             }
28771         }
28772     },
28773
28774     /**
28775      * Returns the number of tabs in this TabPanel.
28776      * @return {Number}
28777      */
28778      getCount : function(){
28779          return this.items.length;
28780      },
28781
28782     /**
28783      * Resizes all the tabs to the passed width
28784      * @param {Number} The new width
28785      */
28786     setTabWidth : function(width){
28787         this.currentTabWidth = width;
28788         for(var i = 0, len = this.items.length; i < len; i++) {
28789                 if(!this.items[i].isHidden()) {
28790                 this.items[i].setWidth(width);
28791             }
28792         }
28793     },
28794
28795     /**
28796      * Destroys this TabPanel
28797      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28798      */
28799     destroy : function(removeEl){
28800         Roo.EventManager.removeResizeListener(this.onResize, this);
28801         for(var i = 0, len = this.items.length; i < len; i++){
28802             this.items[i].purgeListeners();
28803         }
28804         if(removeEl === true){
28805             this.el.update("");
28806             this.el.remove();
28807         }
28808     }
28809 });
28810
28811 /**
28812  * @class Roo.TabPanelItem
28813  * @extends Roo.util.Observable
28814  * Represents an individual item (tab plus body) in a TabPanel.
28815  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28816  * @param {String} id The id of this TabPanelItem
28817  * @param {String} text The text for the tab of this TabPanelItem
28818  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28819  */
28820 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28821     /**
28822      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28823      * @type Roo.TabPanel
28824      */
28825     this.tabPanel = tabPanel;
28826     /**
28827      * The id for this TabPanelItem
28828      * @type String
28829      */
28830     this.id = id;
28831     /** @private */
28832     this.disabled = false;
28833     /** @private */
28834     this.text = text;
28835     /** @private */
28836     this.loaded = false;
28837     this.closable = closable;
28838
28839     /**
28840      * The body element for this TabPanelItem.
28841      * @type Roo.Element
28842      */
28843     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28844     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28845     this.bodyEl.setStyle("display", "block");
28846     this.bodyEl.setStyle("zoom", "1");
28847     this.hideAction();
28848
28849     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28850     /** @private */
28851     this.el = Roo.get(els.el, true);
28852     this.inner = Roo.get(els.inner, true);
28853     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28854     this.pnode = Roo.get(els.el.parentNode, true);
28855     this.el.on("mousedown", this.onTabMouseDown, this);
28856     this.el.on("click", this.onTabClick, this);
28857     /** @private */
28858     if(closable){
28859         var c = Roo.get(els.close, true);
28860         c.dom.title = this.closeText;
28861         c.addClassOnOver("close-over");
28862         c.on("click", this.closeClick, this);
28863      }
28864
28865     this.addEvents({
28866          /**
28867          * @event activate
28868          * Fires when this tab becomes the active tab.
28869          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28870          * @param {Roo.TabPanelItem} this
28871          */
28872         "activate": true,
28873         /**
28874          * @event beforeclose
28875          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28876          * @param {Roo.TabPanelItem} this
28877          * @param {Object} e Set cancel to true on this object to cancel the close.
28878          */
28879         "beforeclose": true,
28880         /**
28881          * @event close
28882          * Fires when this tab is closed.
28883          * @param {Roo.TabPanelItem} this
28884          */
28885          "close": true,
28886         /**
28887          * @event deactivate
28888          * Fires when this tab is no longer the active tab.
28889          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28890          * @param {Roo.TabPanelItem} this
28891          */
28892          "deactivate" : true
28893     });
28894     this.hidden = false;
28895
28896     Roo.TabPanelItem.superclass.constructor.call(this);
28897 };
28898
28899 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28900     purgeListeners : function(){
28901        Roo.util.Observable.prototype.purgeListeners.call(this);
28902        this.el.removeAllListeners();
28903     },
28904     /**
28905      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28906      */
28907     show : function(){
28908         this.pnode.addClass("on");
28909         this.showAction();
28910         if(Roo.isOpera){
28911             this.tabPanel.stripWrap.repaint();
28912         }
28913         this.fireEvent("activate", this.tabPanel, this);
28914     },
28915
28916     /**
28917      * Returns true if this tab is the active tab.
28918      * @return {Boolean}
28919      */
28920     isActive : function(){
28921         return this.tabPanel.getActiveTab() == this;
28922     },
28923
28924     /**
28925      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28926      */
28927     hide : function(){
28928         this.pnode.removeClass("on");
28929         this.hideAction();
28930         this.fireEvent("deactivate", this.tabPanel, this);
28931     },
28932
28933     hideAction : function(){
28934         this.bodyEl.hide();
28935         this.bodyEl.setStyle("position", "absolute");
28936         this.bodyEl.setLeft("-20000px");
28937         this.bodyEl.setTop("-20000px");
28938     },
28939
28940     showAction : function(){
28941         this.bodyEl.setStyle("position", "relative");
28942         this.bodyEl.setTop("");
28943         this.bodyEl.setLeft("");
28944         this.bodyEl.show();
28945     },
28946
28947     /**
28948      * Set the tooltip for the tab.
28949      * @param {String} tooltip The tab's tooltip
28950      */
28951     setTooltip : function(text){
28952         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28953             this.textEl.dom.qtip = text;
28954             this.textEl.dom.removeAttribute('title');
28955         }else{
28956             this.textEl.dom.title = text;
28957         }
28958     },
28959
28960     onTabClick : function(e){
28961         e.preventDefault();
28962         this.tabPanel.activate(this.id);
28963     },
28964
28965     onTabMouseDown : function(e){
28966         e.preventDefault();
28967         this.tabPanel.activate(this.id);
28968     },
28969
28970     getWidth : function(){
28971         return this.inner.getWidth();
28972     },
28973
28974     setWidth : function(width){
28975         var iwidth = width - this.pnode.getPadding("lr");
28976         this.inner.setWidth(iwidth);
28977         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28978         this.pnode.setWidth(width);
28979     },
28980
28981     /**
28982      * Show or hide the tab
28983      * @param {Boolean} hidden True to hide or false to show.
28984      */
28985     setHidden : function(hidden){
28986         this.hidden = hidden;
28987         this.pnode.setStyle("display", hidden ? "none" : "");
28988     },
28989
28990     /**
28991      * Returns true if this tab is "hidden"
28992      * @return {Boolean}
28993      */
28994     isHidden : function(){
28995         return this.hidden;
28996     },
28997
28998     /**
28999      * Returns the text for this tab
29000      * @return {String}
29001      */
29002     getText : function(){
29003         return this.text;
29004     },
29005
29006     autoSize : function(){
29007         //this.el.beginMeasure();
29008         this.textEl.setWidth(1);
29009         /*
29010          *  #2804 [new] Tabs in Roojs
29011          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
29012          */
29013         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
29014         //this.el.endMeasure();
29015     },
29016
29017     /**
29018      * Sets the text for the tab (Note: this also sets the tooltip text)
29019      * @param {String} text The tab's text and tooltip
29020      */
29021     setText : function(text){
29022         this.text = text;
29023         this.textEl.update(text);
29024         this.setTooltip(text);
29025         if(!this.tabPanel.resizeTabs){
29026             this.autoSize();
29027         }
29028     },
29029     /**
29030      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
29031      */
29032     activate : function(){
29033         this.tabPanel.activate(this.id);
29034     },
29035
29036     /**
29037      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
29038      */
29039     disable : function(){
29040         if(this.tabPanel.active != this){
29041             this.disabled = true;
29042             this.pnode.addClass("disabled");
29043         }
29044     },
29045
29046     /**
29047      * Enables this TabPanelItem if it was previously disabled.
29048      */
29049     enable : function(){
29050         this.disabled = false;
29051         this.pnode.removeClass("disabled");
29052     },
29053
29054     /**
29055      * Sets the content for this TabPanelItem.
29056      * @param {String} content The content
29057      * @param {Boolean} loadScripts true to look for and load scripts
29058      */
29059     setContent : function(content, loadScripts){
29060         this.bodyEl.update(content, loadScripts);
29061     },
29062
29063     /**
29064      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29065      * @return {Roo.UpdateManager} The UpdateManager
29066      */
29067     getUpdateManager : function(){
29068         return this.bodyEl.getUpdateManager();
29069     },
29070
29071     /**
29072      * Set a URL to be used to load the content for this TabPanelItem.
29073      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29074      * @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)
29075      * @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)
29076      * @return {Roo.UpdateManager} The UpdateManager
29077      */
29078     setUrl : function(url, params, loadOnce){
29079         if(this.refreshDelegate){
29080             this.un('activate', this.refreshDelegate);
29081         }
29082         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29083         this.on("activate", this.refreshDelegate);
29084         return this.bodyEl.getUpdateManager();
29085     },
29086
29087     /** @private */
29088     _handleRefresh : function(url, params, loadOnce){
29089         if(!loadOnce || !this.loaded){
29090             var updater = this.bodyEl.getUpdateManager();
29091             updater.update(url, params, this._setLoaded.createDelegate(this));
29092         }
29093     },
29094
29095     /**
29096      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29097      *   Will fail silently if the setUrl method has not been called.
29098      *   This does not activate the panel, just updates its content.
29099      */
29100     refresh : function(){
29101         if(this.refreshDelegate){
29102            this.loaded = false;
29103            this.refreshDelegate();
29104         }
29105     },
29106
29107     /** @private */
29108     _setLoaded : function(){
29109         this.loaded = true;
29110     },
29111
29112     /** @private */
29113     closeClick : function(e){
29114         var o = {};
29115         e.stopEvent();
29116         this.fireEvent("beforeclose", this, o);
29117         if(o.cancel !== true){
29118             this.tabPanel.removeTab(this.id);
29119         }
29120     },
29121     /**
29122      * The text displayed in the tooltip for the close icon.
29123      * @type String
29124      */
29125     closeText : "Close this tab"
29126 });
29127
29128 /** @private */
29129 Roo.TabPanel.prototype.createStrip = function(container){
29130     var strip = document.createElement("div");
29131     strip.className = "x-tabs-wrap";
29132     container.appendChild(strip);
29133     return strip;
29134 };
29135 /** @private */
29136 Roo.TabPanel.prototype.createStripList = function(strip){
29137     // div wrapper for retard IE
29138     // returns the "tr" element.
29139     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29140         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29141         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29142     return strip.firstChild.firstChild.firstChild.firstChild;
29143 };
29144 /** @private */
29145 Roo.TabPanel.prototype.createBody = function(container){
29146     var body = document.createElement("div");
29147     Roo.id(body, "tab-body");
29148     Roo.fly(body).addClass("x-tabs-body");
29149     container.appendChild(body);
29150     return body;
29151 };
29152 /** @private */
29153 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29154     var body = Roo.getDom(id);
29155     if(!body){
29156         body = document.createElement("div");
29157         body.id = id;
29158     }
29159     Roo.fly(body).addClass("x-tabs-item-body");
29160     bodyEl.insertBefore(body, bodyEl.firstChild);
29161     return body;
29162 };
29163 /** @private */
29164 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29165     var td = document.createElement("td");
29166     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29167     //stripEl.appendChild(td);
29168     if(closable){
29169         td.className = "x-tabs-closable";
29170         if(!this.closeTpl){
29171             this.closeTpl = new Roo.Template(
29172                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29173                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29174                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29175             );
29176         }
29177         var el = this.closeTpl.overwrite(td, {"text": text});
29178         var close = el.getElementsByTagName("div")[0];
29179         var inner = el.getElementsByTagName("em")[0];
29180         return {"el": el, "close": close, "inner": inner};
29181     } else {
29182         if(!this.tabTpl){
29183             this.tabTpl = new Roo.Template(
29184                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29185                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29186             );
29187         }
29188         var el = this.tabTpl.overwrite(td, {"text": text});
29189         var inner = el.getElementsByTagName("em")[0];
29190         return {"el": el, "inner": inner};
29191     }
29192 };/*
29193  * Based on:
29194  * Ext JS Library 1.1.1
29195  * Copyright(c) 2006-2007, Ext JS, LLC.
29196  *
29197  * Originally Released Under LGPL - original licence link has changed is not relivant.
29198  *
29199  * Fork - LGPL
29200  * <script type="text/javascript">
29201  */
29202
29203 /**
29204  * @class Roo.Button
29205  * @extends Roo.util.Observable
29206  * Simple Button class
29207  * @cfg {String} text The button text
29208  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29209  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29210  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29211  * @cfg {Object} scope The scope of the handler
29212  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29213  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29214  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29215  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29216  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29217  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29218    applies if enableToggle = true)
29219  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29220  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29221   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29222  * @constructor
29223  * Create a new button
29224  * @param {Object} config The config object
29225  */
29226 Roo.Button = function(renderTo, config)
29227 {
29228     if (!config) {
29229         config = renderTo;
29230         renderTo = config.renderTo || false;
29231     }
29232     
29233     Roo.apply(this, config);
29234     this.addEvents({
29235         /**
29236              * @event click
29237              * Fires when this button is clicked
29238              * @param {Button} this
29239              * @param {EventObject} e The click event
29240              */
29241             "click" : true,
29242         /**
29243              * @event toggle
29244              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29245              * @param {Button} this
29246              * @param {Boolean} pressed
29247              */
29248             "toggle" : true,
29249         /**
29250              * @event mouseover
29251              * Fires when the mouse hovers over the button
29252              * @param {Button} this
29253              * @param {Event} e The event object
29254              */
29255         'mouseover' : true,
29256         /**
29257              * @event mouseout
29258              * Fires when the mouse exits the button
29259              * @param {Button} this
29260              * @param {Event} e The event object
29261              */
29262         'mouseout': true,
29263          /**
29264              * @event render
29265              * Fires when the button is rendered
29266              * @param {Button} this
29267              */
29268         'render': true
29269     });
29270     if(this.menu){
29271         this.menu = Roo.menu.MenuMgr.get(this.menu);
29272     }
29273     // register listeners first!!  - so render can be captured..
29274     Roo.util.Observable.call(this);
29275     if(renderTo){
29276         this.render(renderTo);
29277     }
29278     
29279   
29280 };
29281
29282 Roo.extend(Roo.Button, Roo.util.Observable, {
29283     /**
29284      * 
29285      */
29286     
29287     /**
29288      * Read-only. True if this button is hidden
29289      * @type Boolean
29290      */
29291     hidden : false,
29292     /**
29293      * Read-only. True if this button is disabled
29294      * @type Boolean
29295      */
29296     disabled : false,
29297     /**
29298      * Read-only. True if this button is pressed (only if enableToggle = true)
29299      * @type Boolean
29300      */
29301     pressed : false,
29302
29303     /**
29304      * @cfg {Number} tabIndex 
29305      * The DOM tabIndex for this button (defaults to undefined)
29306      */
29307     tabIndex : undefined,
29308
29309     /**
29310      * @cfg {Boolean} enableToggle
29311      * True to enable pressed/not pressed toggling (defaults to false)
29312      */
29313     enableToggle: false,
29314     /**
29315      * @cfg {Mixed} menu
29316      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29317      */
29318     menu : undefined,
29319     /**
29320      * @cfg {String} menuAlign
29321      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29322      */
29323     menuAlign : "tl-bl?",
29324
29325     /**
29326      * @cfg {String} iconCls
29327      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29328      */
29329     iconCls : undefined,
29330     /**
29331      * @cfg {String} type
29332      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29333      */
29334     type : 'button',
29335
29336     // private
29337     menuClassTarget: 'tr',
29338
29339     /**
29340      * @cfg {String} clickEvent
29341      * The type of event to map to the button's event handler (defaults to 'click')
29342      */
29343     clickEvent : 'click',
29344
29345     /**
29346      * @cfg {Boolean} handleMouseEvents
29347      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29348      */
29349     handleMouseEvents : true,
29350
29351     /**
29352      * @cfg {String} tooltipType
29353      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29354      */
29355     tooltipType : 'qtip',
29356
29357     /**
29358      * @cfg {String} cls
29359      * A CSS class to apply to the button's main element.
29360      */
29361     
29362     /**
29363      * @cfg {Roo.Template} template (Optional)
29364      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29365      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29366      * require code modifications if required elements (e.g. a button) aren't present.
29367      */
29368
29369     // private
29370     render : function(renderTo){
29371         var btn;
29372         if(this.hideParent){
29373             this.parentEl = Roo.get(renderTo);
29374         }
29375         if(!this.dhconfig){
29376             if(!this.template){
29377                 if(!Roo.Button.buttonTemplate){
29378                     // hideous table template
29379                     Roo.Button.buttonTemplate = new Roo.Template(
29380                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29381                         '<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>',
29382                         "</tr></tbody></table>");
29383                 }
29384                 this.template = Roo.Button.buttonTemplate;
29385             }
29386             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29387             var btnEl = btn.child("button:first");
29388             btnEl.on('focus', this.onFocus, this);
29389             btnEl.on('blur', this.onBlur, this);
29390             if(this.cls){
29391                 btn.addClass(this.cls);
29392             }
29393             if(this.icon){
29394                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29395             }
29396             if(this.iconCls){
29397                 btnEl.addClass(this.iconCls);
29398                 if(!this.cls){
29399                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29400                 }
29401             }
29402             if(this.tabIndex !== undefined){
29403                 btnEl.dom.tabIndex = this.tabIndex;
29404             }
29405             if(this.tooltip){
29406                 if(typeof this.tooltip == 'object'){
29407                     Roo.QuickTips.tips(Roo.apply({
29408                           target: btnEl.id
29409                     }, this.tooltip));
29410                 } else {
29411                     btnEl.dom[this.tooltipType] = this.tooltip;
29412                 }
29413             }
29414         }else{
29415             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29416         }
29417         this.el = btn;
29418         if(this.id){
29419             this.el.dom.id = this.el.id = this.id;
29420         }
29421         if(this.menu){
29422             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29423             this.menu.on("show", this.onMenuShow, this);
29424             this.menu.on("hide", this.onMenuHide, this);
29425         }
29426         btn.addClass("x-btn");
29427         if(Roo.isIE && !Roo.isIE7){
29428             this.autoWidth.defer(1, this);
29429         }else{
29430             this.autoWidth();
29431         }
29432         if(this.handleMouseEvents){
29433             btn.on("mouseover", this.onMouseOver, this);
29434             btn.on("mouseout", this.onMouseOut, this);
29435             btn.on("mousedown", this.onMouseDown, this);
29436         }
29437         btn.on(this.clickEvent, this.onClick, this);
29438         //btn.on("mouseup", this.onMouseUp, this);
29439         if(this.hidden){
29440             this.hide();
29441         }
29442         if(this.disabled){
29443             this.disable();
29444         }
29445         Roo.ButtonToggleMgr.register(this);
29446         if(this.pressed){
29447             this.el.addClass("x-btn-pressed");
29448         }
29449         if(this.repeat){
29450             var repeater = new Roo.util.ClickRepeater(btn,
29451                 typeof this.repeat == "object" ? this.repeat : {}
29452             );
29453             repeater.on("click", this.onClick,  this);
29454         }
29455         
29456         this.fireEvent('render', this);
29457         
29458     },
29459     /**
29460      * Returns the button's underlying element
29461      * @return {Roo.Element} The element
29462      */
29463     getEl : function(){
29464         return this.el;  
29465     },
29466     
29467     /**
29468      * Destroys this Button and removes any listeners.
29469      */
29470     destroy : function(){
29471         Roo.ButtonToggleMgr.unregister(this);
29472         this.el.removeAllListeners();
29473         this.purgeListeners();
29474         this.el.remove();
29475     },
29476
29477     // private
29478     autoWidth : function(){
29479         if(this.el){
29480             this.el.setWidth("auto");
29481             if(Roo.isIE7 && Roo.isStrict){
29482                 var ib = this.el.child('button');
29483                 if(ib && ib.getWidth() > 20){
29484                     ib.clip();
29485                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29486                 }
29487             }
29488             if(this.minWidth){
29489                 if(this.hidden){
29490                     this.el.beginMeasure();
29491                 }
29492                 if(this.el.getWidth() < this.minWidth){
29493                     this.el.setWidth(this.minWidth);
29494                 }
29495                 if(this.hidden){
29496                     this.el.endMeasure();
29497                 }
29498             }
29499         }
29500     },
29501
29502     /**
29503      * Assigns this button's click handler
29504      * @param {Function} handler The function to call when the button is clicked
29505      * @param {Object} scope (optional) Scope for the function passed in
29506      */
29507     setHandler : function(handler, scope){
29508         this.handler = handler;
29509         this.scope = scope;  
29510     },
29511     
29512     /**
29513      * Sets this button's text
29514      * @param {String} text The button text
29515      */
29516     setText : function(text){
29517         this.text = text;
29518         if(this.el){
29519             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29520         }
29521         this.autoWidth();
29522     },
29523     
29524     /**
29525      * Gets the text for this button
29526      * @return {String} The button text
29527      */
29528     getText : function(){
29529         return this.text;  
29530     },
29531     
29532     /**
29533      * Show this button
29534      */
29535     show: function(){
29536         this.hidden = false;
29537         if(this.el){
29538             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29539         }
29540     },
29541     
29542     /**
29543      * Hide this button
29544      */
29545     hide: function(){
29546         this.hidden = true;
29547         if(this.el){
29548             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29549         }
29550     },
29551     
29552     /**
29553      * Convenience function for boolean show/hide
29554      * @param {Boolean} visible True to show, false to hide
29555      */
29556     setVisible: function(visible){
29557         if(visible) {
29558             this.show();
29559         }else{
29560             this.hide();
29561         }
29562     },
29563     
29564     /**
29565      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29566      * @param {Boolean} state (optional) Force a particular state
29567      */
29568     toggle : function(state){
29569         state = state === undefined ? !this.pressed : state;
29570         if(state != this.pressed){
29571             if(state){
29572                 this.el.addClass("x-btn-pressed");
29573                 this.pressed = true;
29574                 this.fireEvent("toggle", this, true);
29575             }else{
29576                 this.el.removeClass("x-btn-pressed");
29577                 this.pressed = false;
29578                 this.fireEvent("toggle", this, false);
29579             }
29580             if(this.toggleHandler){
29581                 this.toggleHandler.call(this.scope || this, this, state);
29582             }
29583         }
29584     },
29585     
29586     /**
29587      * Focus the button
29588      */
29589     focus : function(){
29590         this.el.child('button:first').focus();
29591     },
29592     
29593     /**
29594      * Disable this button
29595      */
29596     disable : function(){
29597         if(this.el){
29598             this.el.addClass("x-btn-disabled");
29599         }
29600         this.disabled = true;
29601     },
29602     
29603     /**
29604      * Enable this button
29605      */
29606     enable : function(){
29607         if(this.el){
29608             this.el.removeClass("x-btn-disabled");
29609         }
29610         this.disabled = false;
29611     },
29612
29613     /**
29614      * Convenience function for boolean enable/disable
29615      * @param {Boolean} enabled True to enable, false to disable
29616      */
29617     setDisabled : function(v){
29618         this[v !== true ? "enable" : "disable"]();
29619     },
29620
29621     // private
29622     onClick : function(e)
29623     {
29624         if(e){
29625             e.preventDefault();
29626         }
29627         if(e.button != 0){
29628             return;
29629         }
29630         if(!this.disabled){
29631             if(this.enableToggle){
29632                 this.toggle();
29633             }
29634             if(this.menu && !this.menu.isVisible()){
29635                 this.menu.show(this.el, this.menuAlign);
29636             }
29637             this.fireEvent("click", this, e);
29638             if(this.handler){
29639                 this.el.removeClass("x-btn-over");
29640                 this.handler.call(this.scope || this, this, e);
29641             }
29642         }
29643     },
29644     // private
29645     onMouseOver : function(e){
29646         if(!this.disabled){
29647             this.el.addClass("x-btn-over");
29648             this.fireEvent('mouseover', this, e);
29649         }
29650     },
29651     // private
29652     onMouseOut : function(e){
29653         if(!e.within(this.el,  true)){
29654             this.el.removeClass("x-btn-over");
29655             this.fireEvent('mouseout', this, e);
29656         }
29657     },
29658     // private
29659     onFocus : function(e){
29660         if(!this.disabled){
29661             this.el.addClass("x-btn-focus");
29662         }
29663     },
29664     // private
29665     onBlur : function(e){
29666         this.el.removeClass("x-btn-focus");
29667     },
29668     // private
29669     onMouseDown : function(e){
29670         if(!this.disabled && e.button == 0){
29671             this.el.addClass("x-btn-click");
29672             Roo.get(document).on('mouseup', this.onMouseUp, this);
29673         }
29674     },
29675     // private
29676     onMouseUp : function(e){
29677         if(e.button == 0){
29678             this.el.removeClass("x-btn-click");
29679             Roo.get(document).un('mouseup', this.onMouseUp, this);
29680         }
29681     },
29682     // private
29683     onMenuShow : function(e){
29684         this.el.addClass("x-btn-menu-active");
29685     },
29686     // private
29687     onMenuHide : function(e){
29688         this.el.removeClass("x-btn-menu-active");
29689     }   
29690 });
29691
29692 // Private utility class used by Button
29693 Roo.ButtonToggleMgr = function(){
29694    var groups = {};
29695    
29696    function toggleGroup(btn, state){
29697        if(state){
29698            var g = groups[btn.toggleGroup];
29699            for(var i = 0, l = g.length; i < l; i++){
29700                if(g[i] != btn){
29701                    g[i].toggle(false);
29702                }
29703            }
29704        }
29705    }
29706    
29707    return {
29708        register : function(btn){
29709            if(!btn.toggleGroup){
29710                return;
29711            }
29712            var g = groups[btn.toggleGroup];
29713            if(!g){
29714                g = groups[btn.toggleGroup] = [];
29715            }
29716            g.push(btn);
29717            btn.on("toggle", toggleGroup);
29718        },
29719        
29720        unregister : function(btn){
29721            if(!btn.toggleGroup){
29722                return;
29723            }
29724            var g = groups[btn.toggleGroup];
29725            if(g){
29726                g.remove(btn);
29727                btn.un("toggle", toggleGroup);
29728            }
29729        }
29730    };
29731 }();/*
29732  * Based on:
29733  * Ext JS Library 1.1.1
29734  * Copyright(c) 2006-2007, Ext JS, LLC.
29735  *
29736  * Originally Released Under LGPL - original licence link has changed is not relivant.
29737  *
29738  * Fork - LGPL
29739  * <script type="text/javascript">
29740  */
29741  
29742 /**
29743  * @class Roo.SplitButton
29744  * @extends Roo.Button
29745  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29746  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29747  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29748  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29749  * @cfg {String} arrowTooltip The title attribute of the arrow
29750  * @constructor
29751  * Create a new menu button
29752  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29753  * @param {Object} config The config object
29754  */
29755 Roo.SplitButton = function(renderTo, config){
29756     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29757     /**
29758      * @event arrowclick
29759      * Fires when this button's arrow is clicked
29760      * @param {SplitButton} this
29761      * @param {EventObject} e The click event
29762      */
29763     this.addEvents({"arrowclick":true});
29764 };
29765
29766 Roo.extend(Roo.SplitButton, Roo.Button, {
29767     render : function(renderTo){
29768         // this is one sweet looking template!
29769         var tpl = new Roo.Template(
29770             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29771             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29772             '<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>',
29773             "</tbody></table></td><td>",
29774             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29775             '<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>',
29776             "</tbody></table></td></tr></table>"
29777         );
29778         var btn = tpl.append(renderTo, [this.text, this.type], true);
29779         var btnEl = btn.child("button");
29780         if(this.cls){
29781             btn.addClass(this.cls);
29782         }
29783         if(this.icon){
29784             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29785         }
29786         if(this.iconCls){
29787             btnEl.addClass(this.iconCls);
29788             if(!this.cls){
29789                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29790             }
29791         }
29792         this.el = btn;
29793         if(this.handleMouseEvents){
29794             btn.on("mouseover", this.onMouseOver, this);
29795             btn.on("mouseout", this.onMouseOut, this);
29796             btn.on("mousedown", this.onMouseDown, this);
29797             btn.on("mouseup", this.onMouseUp, this);
29798         }
29799         btn.on(this.clickEvent, this.onClick, this);
29800         if(this.tooltip){
29801             if(typeof this.tooltip == 'object'){
29802                 Roo.QuickTips.tips(Roo.apply({
29803                       target: btnEl.id
29804                 }, this.tooltip));
29805             } else {
29806                 btnEl.dom[this.tooltipType] = this.tooltip;
29807             }
29808         }
29809         if(this.arrowTooltip){
29810             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29811         }
29812         if(this.hidden){
29813             this.hide();
29814         }
29815         if(this.disabled){
29816             this.disable();
29817         }
29818         if(this.pressed){
29819             this.el.addClass("x-btn-pressed");
29820         }
29821         if(Roo.isIE && !Roo.isIE7){
29822             this.autoWidth.defer(1, this);
29823         }else{
29824             this.autoWidth();
29825         }
29826         if(this.menu){
29827             this.menu.on("show", this.onMenuShow, this);
29828             this.menu.on("hide", this.onMenuHide, this);
29829         }
29830         this.fireEvent('render', this);
29831     },
29832
29833     // private
29834     autoWidth : function(){
29835         if(this.el){
29836             var tbl = this.el.child("table:first");
29837             var tbl2 = this.el.child("table:last");
29838             this.el.setWidth("auto");
29839             tbl.setWidth("auto");
29840             if(Roo.isIE7 && Roo.isStrict){
29841                 var ib = this.el.child('button:first');
29842                 if(ib && ib.getWidth() > 20){
29843                     ib.clip();
29844                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29845                 }
29846             }
29847             if(this.minWidth){
29848                 if(this.hidden){
29849                     this.el.beginMeasure();
29850                 }
29851                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29852                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29853                 }
29854                 if(this.hidden){
29855                     this.el.endMeasure();
29856                 }
29857             }
29858             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29859         } 
29860     },
29861     /**
29862      * Sets this button's click handler
29863      * @param {Function} handler The function to call when the button is clicked
29864      * @param {Object} scope (optional) Scope for the function passed above
29865      */
29866     setHandler : function(handler, scope){
29867         this.handler = handler;
29868         this.scope = scope;  
29869     },
29870     
29871     /**
29872      * Sets this button's arrow click handler
29873      * @param {Function} handler The function to call when the arrow is clicked
29874      * @param {Object} scope (optional) Scope for the function passed above
29875      */
29876     setArrowHandler : function(handler, scope){
29877         this.arrowHandler = handler;
29878         this.scope = scope;  
29879     },
29880     
29881     /**
29882      * Focus the button
29883      */
29884     focus : function(){
29885         if(this.el){
29886             this.el.child("button:first").focus();
29887         }
29888     },
29889
29890     // private
29891     onClick : function(e){
29892         e.preventDefault();
29893         if(!this.disabled){
29894             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29895                 if(this.menu && !this.menu.isVisible()){
29896                     this.menu.show(this.el, this.menuAlign);
29897                 }
29898                 this.fireEvent("arrowclick", this, e);
29899                 if(this.arrowHandler){
29900                     this.arrowHandler.call(this.scope || this, this, e);
29901                 }
29902             }else{
29903                 this.fireEvent("click", this, e);
29904                 if(this.handler){
29905                     this.handler.call(this.scope || this, this, e);
29906                 }
29907             }
29908         }
29909     },
29910     // private
29911     onMouseDown : function(e){
29912         if(!this.disabled){
29913             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29914         }
29915     },
29916     // private
29917     onMouseUp : function(e){
29918         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29919     }   
29920 });
29921
29922
29923 // backwards compat
29924 Roo.MenuButton = Roo.SplitButton;/*
29925  * Based on:
29926  * Ext JS Library 1.1.1
29927  * Copyright(c) 2006-2007, Ext JS, LLC.
29928  *
29929  * Originally Released Under LGPL - original licence link has changed is not relivant.
29930  *
29931  * Fork - LGPL
29932  * <script type="text/javascript">
29933  */
29934
29935 /**
29936  * @class Roo.Toolbar
29937  * Basic Toolbar class.
29938  * @constructor
29939  * Creates a new Toolbar
29940  * @param {Object} container The config object
29941  */ 
29942 Roo.Toolbar = function(container, buttons, config)
29943 {
29944     /// old consturctor format still supported..
29945     if(container instanceof Array){ // omit the container for later rendering
29946         buttons = container;
29947         config = buttons;
29948         container = null;
29949     }
29950     if (typeof(container) == 'object' && container.xtype) {
29951         config = container;
29952         container = config.container;
29953         buttons = config.buttons || []; // not really - use items!!
29954     }
29955     var xitems = [];
29956     if (config && config.items) {
29957         xitems = config.items;
29958         delete config.items;
29959     }
29960     Roo.apply(this, config);
29961     this.buttons = buttons;
29962     
29963     if(container){
29964         this.render(container);
29965     }
29966     this.xitems = xitems;
29967     Roo.each(xitems, function(b) {
29968         this.add(b);
29969     }, this);
29970     
29971 };
29972
29973 Roo.Toolbar.prototype = {
29974     /**
29975      * @cfg {Array} items
29976      * array of button configs or elements to add (will be converted to a MixedCollection)
29977      */
29978     
29979     /**
29980      * @cfg {String/HTMLElement/Element} container
29981      * The id or element that will contain the toolbar
29982      */
29983     // private
29984     render : function(ct){
29985         this.el = Roo.get(ct);
29986         if(this.cls){
29987             this.el.addClass(this.cls);
29988         }
29989         // using a table allows for vertical alignment
29990         // 100% width is needed by Safari...
29991         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29992         this.tr = this.el.child("tr", true);
29993         var autoId = 0;
29994         this.items = new Roo.util.MixedCollection(false, function(o){
29995             return o.id || ("item" + (++autoId));
29996         });
29997         if(this.buttons){
29998             this.add.apply(this, this.buttons);
29999             delete this.buttons;
30000         }
30001     },
30002
30003     /**
30004      * Adds element(s) to the toolbar -- this function takes a variable number of 
30005      * arguments of mixed type and adds them to the toolbar.
30006      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
30007      * <ul>
30008      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
30009      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
30010      * <li>Field: Any form field (equivalent to {@link #addField})</li>
30011      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
30012      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
30013      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
30014      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
30015      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
30016      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
30017      * </ul>
30018      * @param {Mixed} arg2
30019      * @param {Mixed} etc.
30020      */
30021     add : function(){
30022         var a = arguments, l = a.length;
30023         for(var i = 0; i < l; i++){
30024             this._add(a[i]);
30025         }
30026     },
30027     // private..
30028     _add : function(el) {
30029         
30030         if (el.xtype) {
30031             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
30032         }
30033         
30034         if (el.applyTo){ // some kind of form field
30035             return this.addField(el);
30036         } 
30037         if (el.render){ // some kind of Toolbar.Item
30038             return this.addItem(el);
30039         }
30040         if (typeof el == "string"){ // string
30041             if(el == "separator" || el == "-"){
30042                 return this.addSeparator();
30043             }
30044             if (el == " "){
30045                 return this.addSpacer();
30046             }
30047             if(el == "->"){
30048                 return this.addFill();
30049             }
30050             return this.addText(el);
30051             
30052         }
30053         if(el.tagName){ // element
30054             return this.addElement(el);
30055         }
30056         if(typeof el == "object"){ // must be button config?
30057             return this.addButton(el);
30058         }
30059         // and now what?!?!
30060         return false;
30061         
30062     },
30063     
30064     /**
30065      * Add an Xtype element
30066      * @param {Object} xtype Xtype Object
30067      * @return {Object} created Object
30068      */
30069     addxtype : function(e){
30070         return this.add(e);  
30071     },
30072     
30073     /**
30074      * Returns the Element for this toolbar.
30075      * @return {Roo.Element}
30076      */
30077     getEl : function(){
30078         return this.el;  
30079     },
30080     
30081     /**
30082      * Adds a separator
30083      * @return {Roo.Toolbar.Item} The separator item
30084      */
30085     addSeparator : function(){
30086         return this.addItem(new Roo.Toolbar.Separator());
30087     },
30088
30089     /**
30090      * Adds a spacer element
30091      * @return {Roo.Toolbar.Spacer} The spacer item
30092      */
30093     addSpacer : function(){
30094         return this.addItem(new Roo.Toolbar.Spacer());
30095     },
30096
30097     /**
30098      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30099      * @return {Roo.Toolbar.Fill} The fill item
30100      */
30101     addFill : function(){
30102         return this.addItem(new Roo.Toolbar.Fill());
30103     },
30104
30105     /**
30106      * Adds any standard HTML element to the toolbar
30107      * @param {String/HTMLElement/Element} el The element or id of the element to add
30108      * @return {Roo.Toolbar.Item} The element's item
30109      */
30110     addElement : function(el){
30111         return this.addItem(new Roo.Toolbar.Item(el));
30112     },
30113     /**
30114      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30115      * @type Roo.util.MixedCollection  
30116      */
30117     items : false,
30118      
30119     /**
30120      * Adds any Toolbar.Item or subclass
30121      * @param {Roo.Toolbar.Item} item
30122      * @return {Roo.Toolbar.Item} The item
30123      */
30124     addItem : function(item){
30125         var td = this.nextBlock();
30126         item.render(td);
30127         this.items.add(item);
30128         return item;
30129     },
30130     
30131     /**
30132      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30133      * @param {Object/Array} config A button config or array of configs
30134      * @return {Roo.Toolbar.Button/Array}
30135      */
30136     addButton : function(config){
30137         if(config instanceof Array){
30138             var buttons = [];
30139             for(var i = 0, len = config.length; i < len; i++) {
30140                 buttons.push(this.addButton(config[i]));
30141             }
30142             return buttons;
30143         }
30144         var b = config;
30145         if(!(config instanceof Roo.Toolbar.Button)){
30146             b = config.split ?
30147                 new Roo.Toolbar.SplitButton(config) :
30148                 new Roo.Toolbar.Button(config);
30149         }
30150         var td = this.nextBlock();
30151         b.render(td);
30152         this.items.add(b);
30153         return b;
30154     },
30155     
30156     /**
30157      * Adds text to the toolbar
30158      * @param {String} text The text to add
30159      * @return {Roo.Toolbar.Item} The element's item
30160      */
30161     addText : function(text){
30162         return this.addItem(new Roo.Toolbar.TextItem(text));
30163     },
30164     
30165     /**
30166      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30167      * @param {Number} index The index where the item is to be inserted
30168      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30169      * @return {Roo.Toolbar.Button/Item}
30170      */
30171     insertButton : function(index, item){
30172         if(item instanceof Array){
30173             var buttons = [];
30174             for(var i = 0, len = item.length; i < len; i++) {
30175                buttons.push(this.insertButton(index + i, item[i]));
30176             }
30177             return buttons;
30178         }
30179         if (!(item instanceof Roo.Toolbar.Button)){
30180            item = new Roo.Toolbar.Button(item);
30181         }
30182         var td = document.createElement("td");
30183         this.tr.insertBefore(td, this.tr.childNodes[index]);
30184         item.render(td);
30185         this.items.insert(index, item);
30186         return item;
30187     },
30188     
30189     /**
30190      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30191      * @param {Object} config
30192      * @return {Roo.Toolbar.Item} The element's item
30193      */
30194     addDom : function(config, returnEl){
30195         var td = this.nextBlock();
30196         Roo.DomHelper.overwrite(td, config);
30197         var ti = new Roo.Toolbar.Item(td.firstChild);
30198         ti.render(td);
30199         this.items.add(ti);
30200         return ti;
30201     },
30202
30203     /**
30204      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30205      * @type Roo.util.MixedCollection  
30206      */
30207     fields : false,
30208     
30209     /**
30210      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30211      * Note: the field should not have been rendered yet. For a field that has already been
30212      * rendered, use {@link #addElement}.
30213      * @param {Roo.form.Field} field
30214      * @return {Roo.ToolbarItem}
30215      */
30216      
30217       
30218     addField : function(field) {
30219         if (!this.fields) {
30220             var autoId = 0;
30221             this.fields = new Roo.util.MixedCollection(false, function(o){
30222                 return o.id || ("item" + (++autoId));
30223             });
30224
30225         }
30226         
30227         var td = this.nextBlock();
30228         field.render(td);
30229         var ti = new Roo.Toolbar.Item(td.firstChild);
30230         ti.render(td);
30231         this.items.add(ti);
30232         this.fields.add(field);
30233         return ti;
30234     },
30235     /**
30236      * Hide the toolbar
30237      * @method hide
30238      */
30239      
30240       
30241     hide : function()
30242     {
30243         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30244         this.el.child('div').hide();
30245     },
30246     /**
30247      * Show the toolbar
30248      * @method show
30249      */
30250     show : function()
30251     {
30252         this.el.child('div').show();
30253     },
30254       
30255     // private
30256     nextBlock : function(){
30257         var td = document.createElement("td");
30258         this.tr.appendChild(td);
30259         return td;
30260     },
30261
30262     // private
30263     destroy : function(){
30264         if(this.items){ // rendered?
30265             Roo.destroy.apply(Roo, this.items.items);
30266         }
30267         if(this.fields){ // rendered?
30268             Roo.destroy.apply(Roo, this.fields.items);
30269         }
30270         Roo.Element.uncache(this.el, this.tr);
30271     }
30272 };
30273
30274 /**
30275  * @class Roo.Toolbar.Item
30276  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30277  * @constructor
30278  * Creates a new Item
30279  * @param {HTMLElement} el 
30280  */
30281 Roo.Toolbar.Item = function(el){
30282     var cfg = {};
30283     if (typeof (el.xtype) != 'undefined') {
30284         cfg = el;
30285         el = cfg.el;
30286     }
30287     
30288     this.el = Roo.getDom(el);
30289     this.id = Roo.id(this.el);
30290     this.hidden = false;
30291     
30292     this.addEvents({
30293          /**
30294              * @event render
30295              * Fires when the button is rendered
30296              * @param {Button} this
30297              */
30298         'render': true
30299     });
30300     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30301 };
30302 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30303 //Roo.Toolbar.Item.prototype = {
30304     
30305     /**
30306      * Get this item's HTML Element
30307      * @return {HTMLElement}
30308      */
30309     getEl : function(){
30310        return this.el;  
30311     },
30312
30313     // private
30314     render : function(td){
30315         
30316          this.td = td;
30317         td.appendChild(this.el);
30318         
30319         this.fireEvent('render', this);
30320     },
30321     
30322     /**
30323      * Removes and destroys this item.
30324      */
30325     destroy : function(){
30326         this.td.parentNode.removeChild(this.td);
30327     },
30328     
30329     /**
30330      * Shows this item.
30331      */
30332     show: function(){
30333         this.hidden = false;
30334         this.td.style.display = "";
30335     },
30336     
30337     /**
30338      * Hides this item.
30339      */
30340     hide: function(){
30341         this.hidden = true;
30342         this.td.style.display = "none";
30343     },
30344     
30345     /**
30346      * Convenience function for boolean show/hide.
30347      * @param {Boolean} visible true to show/false to hide
30348      */
30349     setVisible: function(visible){
30350         if(visible) {
30351             this.show();
30352         }else{
30353             this.hide();
30354         }
30355     },
30356     
30357     /**
30358      * Try to focus this item.
30359      */
30360     focus : function(){
30361         Roo.fly(this.el).focus();
30362     },
30363     
30364     /**
30365      * Disables this item.
30366      */
30367     disable : function(){
30368         Roo.fly(this.td).addClass("x-item-disabled");
30369         this.disabled = true;
30370         this.el.disabled = true;
30371     },
30372     
30373     /**
30374      * Enables this item.
30375      */
30376     enable : function(){
30377         Roo.fly(this.td).removeClass("x-item-disabled");
30378         this.disabled = false;
30379         this.el.disabled = false;
30380     }
30381 });
30382
30383
30384 /**
30385  * @class Roo.Toolbar.Separator
30386  * @extends Roo.Toolbar.Item
30387  * A simple toolbar separator class
30388  * @constructor
30389  * Creates a new Separator
30390  */
30391 Roo.Toolbar.Separator = function(cfg){
30392     
30393     var s = document.createElement("span");
30394     s.className = "ytb-sep";
30395     if (cfg) {
30396         cfg.el = s;
30397     }
30398     
30399     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30400 };
30401 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30402     enable:Roo.emptyFn,
30403     disable:Roo.emptyFn,
30404     focus:Roo.emptyFn
30405 });
30406
30407 /**
30408  * @class Roo.Toolbar.Spacer
30409  * @extends Roo.Toolbar.Item
30410  * A simple element that adds extra horizontal space to a toolbar.
30411  * @constructor
30412  * Creates a new Spacer
30413  */
30414 Roo.Toolbar.Spacer = function(cfg){
30415     var s = document.createElement("div");
30416     s.className = "ytb-spacer";
30417     if (cfg) {
30418         cfg.el = s;
30419     }
30420     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30421 };
30422 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30423     enable:Roo.emptyFn,
30424     disable:Roo.emptyFn,
30425     focus:Roo.emptyFn
30426 });
30427
30428 /**
30429  * @class Roo.Toolbar.Fill
30430  * @extends Roo.Toolbar.Spacer
30431  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30432  * @constructor
30433  * Creates a new Spacer
30434  */
30435 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30436     // private
30437     render : function(td){
30438         td.style.width = '100%';
30439         Roo.Toolbar.Fill.superclass.render.call(this, td);
30440     }
30441 });
30442
30443 /**
30444  * @class Roo.Toolbar.TextItem
30445  * @extends Roo.Toolbar.Item
30446  * A simple class that renders text directly into a toolbar.
30447  * @constructor
30448  * Creates a new TextItem
30449  * @cfg {string} text 
30450  */
30451 Roo.Toolbar.TextItem = function(cfg){
30452     var  text = cfg || "";
30453     if (typeof(cfg) == 'object') {
30454         text = cfg.text || "";
30455     }  else {
30456         cfg = null;
30457     }
30458     var s = document.createElement("span");
30459     s.className = "ytb-text";
30460     s.innerHTML = text;
30461     if (cfg) {
30462         cfg.el  = s;
30463     }
30464     
30465     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30466 };
30467 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30468     
30469      
30470     enable:Roo.emptyFn,
30471     disable:Roo.emptyFn,
30472     focus:Roo.emptyFn
30473 });
30474
30475 /**
30476  * @class Roo.Toolbar.Button
30477  * @extends Roo.Button
30478  * A button that renders into a toolbar.
30479  * @constructor
30480  * Creates a new Button
30481  * @param {Object} config A standard {@link Roo.Button} config object
30482  */
30483 Roo.Toolbar.Button = function(config){
30484     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30485 };
30486 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30487     render : function(td){
30488         this.td = td;
30489         Roo.Toolbar.Button.superclass.render.call(this, td);
30490     },
30491     
30492     /**
30493      * Removes and destroys this button
30494      */
30495     destroy : function(){
30496         Roo.Toolbar.Button.superclass.destroy.call(this);
30497         this.td.parentNode.removeChild(this.td);
30498     },
30499     
30500     /**
30501      * Shows this button
30502      */
30503     show: function(){
30504         this.hidden = false;
30505         this.td.style.display = "";
30506     },
30507     
30508     /**
30509      * Hides this button
30510      */
30511     hide: function(){
30512         this.hidden = true;
30513         this.td.style.display = "none";
30514     },
30515
30516     /**
30517      * Disables this item
30518      */
30519     disable : function(){
30520         Roo.fly(this.td).addClass("x-item-disabled");
30521         this.disabled = true;
30522     },
30523
30524     /**
30525      * Enables this item
30526      */
30527     enable : function(){
30528         Roo.fly(this.td).removeClass("x-item-disabled");
30529         this.disabled = false;
30530     }
30531 });
30532 // backwards compat
30533 Roo.ToolbarButton = Roo.Toolbar.Button;
30534
30535 /**
30536  * @class Roo.Toolbar.SplitButton
30537  * @extends Roo.SplitButton
30538  * A menu button that renders into a toolbar.
30539  * @constructor
30540  * Creates a new SplitButton
30541  * @param {Object} config A standard {@link Roo.SplitButton} config object
30542  */
30543 Roo.Toolbar.SplitButton = function(config){
30544     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30545 };
30546 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30547     render : function(td){
30548         this.td = td;
30549         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30550     },
30551     
30552     /**
30553      * Removes and destroys this button
30554      */
30555     destroy : function(){
30556         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30557         this.td.parentNode.removeChild(this.td);
30558     },
30559     
30560     /**
30561      * Shows this button
30562      */
30563     show: function(){
30564         this.hidden = false;
30565         this.td.style.display = "";
30566     },
30567     
30568     /**
30569      * Hides this button
30570      */
30571     hide: function(){
30572         this.hidden = true;
30573         this.td.style.display = "none";
30574     }
30575 });
30576
30577 // backwards compat
30578 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30579  * Based on:
30580  * Ext JS Library 1.1.1
30581  * Copyright(c) 2006-2007, Ext JS, LLC.
30582  *
30583  * Originally Released Under LGPL - original licence link has changed is not relivant.
30584  *
30585  * Fork - LGPL
30586  * <script type="text/javascript">
30587  */
30588  
30589 /**
30590  * @class Roo.PagingToolbar
30591  * @extends Roo.Toolbar
30592  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30593  * @constructor
30594  * Create a new PagingToolbar
30595  * @param {Object} config The config object
30596  */
30597 Roo.PagingToolbar = function(el, ds, config)
30598 {
30599     // old args format still supported... - xtype is prefered..
30600     if (typeof(el) == 'object' && el.xtype) {
30601         // created from xtype...
30602         config = el;
30603         ds = el.dataSource;
30604         el = config.container;
30605     }
30606     var items = [];
30607     if (config.items) {
30608         items = config.items;
30609         config.items = [];
30610     }
30611     
30612     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30613     this.ds = ds;
30614     this.cursor = 0;
30615     this.renderButtons(this.el);
30616     this.bind(ds);
30617     
30618     // supprot items array.
30619    
30620     Roo.each(items, function(e) {
30621         this.add(Roo.factory(e));
30622     },this);
30623     
30624 };
30625
30626 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30627     /**
30628      * @cfg {Roo.data.Store} dataSource
30629      * The underlying data store providing the paged data
30630      */
30631     /**
30632      * @cfg {String/HTMLElement/Element} container
30633      * container The id or element that will contain the toolbar
30634      */
30635     /**
30636      * @cfg {Boolean} displayInfo
30637      * True to display the displayMsg (defaults to false)
30638      */
30639     /**
30640      * @cfg {Number} pageSize
30641      * The number of records to display per page (defaults to 20)
30642      */
30643     pageSize: 20,
30644     /**
30645      * @cfg {String} displayMsg
30646      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30647      */
30648     displayMsg : 'Displaying {0} - {1} of {2}',
30649     /**
30650      * @cfg {String} emptyMsg
30651      * The message to display when no records are found (defaults to "No data to display")
30652      */
30653     emptyMsg : 'No data to display',
30654     /**
30655      * Customizable piece of the default paging text (defaults to "Page")
30656      * @type String
30657      */
30658     beforePageText : "Page",
30659     /**
30660      * Customizable piece of the default paging text (defaults to "of %0")
30661      * @type String
30662      */
30663     afterPageText : "of {0}",
30664     /**
30665      * Customizable piece of the default paging text (defaults to "First Page")
30666      * @type String
30667      */
30668     firstText : "First Page",
30669     /**
30670      * Customizable piece of the default paging text (defaults to "Previous Page")
30671      * @type String
30672      */
30673     prevText : "Previous Page",
30674     /**
30675      * Customizable piece of the default paging text (defaults to "Next Page")
30676      * @type String
30677      */
30678     nextText : "Next Page",
30679     /**
30680      * Customizable piece of the default paging text (defaults to "Last Page")
30681      * @type String
30682      */
30683     lastText : "Last Page",
30684     /**
30685      * Customizable piece of the default paging text (defaults to "Refresh")
30686      * @type String
30687      */
30688     refreshText : "Refresh",
30689
30690     // private
30691     renderButtons : function(el){
30692         Roo.PagingToolbar.superclass.render.call(this, el);
30693         this.first = this.addButton({
30694             tooltip: this.firstText,
30695             cls: "x-btn-icon x-grid-page-first",
30696             disabled: true,
30697             handler: this.onClick.createDelegate(this, ["first"])
30698         });
30699         this.prev = this.addButton({
30700             tooltip: this.prevText,
30701             cls: "x-btn-icon x-grid-page-prev",
30702             disabled: true,
30703             handler: this.onClick.createDelegate(this, ["prev"])
30704         });
30705         //this.addSeparator();
30706         this.add(this.beforePageText);
30707         this.field = Roo.get(this.addDom({
30708            tag: "input",
30709            type: "text",
30710            size: "3",
30711            value: "1",
30712            cls: "x-grid-page-number"
30713         }).el);
30714         this.field.on("keydown", this.onPagingKeydown, this);
30715         this.field.on("focus", function(){this.dom.select();});
30716         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30717         this.field.setHeight(18);
30718         //this.addSeparator();
30719         this.next = this.addButton({
30720             tooltip: this.nextText,
30721             cls: "x-btn-icon x-grid-page-next",
30722             disabled: true,
30723             handler: this.onClick.createDelegate(this, ["next"])
30724         });
30725         this.last = this.addButton({
30726             tooltip: this.lastText,
30727             cls: "x-btn-icon x-grid-page-last",
30728             disabled: true,
30729             handler: this.onClick.createDelegate(this, ["last"])
30730         });
30731         //this.addSeparator();
30732         this.loading = this.addButton({
30733             tooltip: this.refreshText,
30734             cls: "x-btn-icon x-grid-loading",
30735             handler: this.onClick.createDelegate(this, ["refresh"])
30736         });
30737
30738         if(this.displayInfo){
30739             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30740         }
30741     },
30742
30743     // private
30744     updateInfo : function(){
30745         if(this.displayEl){
30746             var count = this.ds.getCount();
30747             var msg = count == 0 ?
30748                 this.emptyMsg :
30749                 String.format(
30750                     this.displayMsg,
30751                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30752                 );
30753             this.displayEl.update(msg);
30754         }
30755     },
30756
30757     // private
30758     onLoad : function(ds, r, o){
30759        this.cursor = o.params ? o.params.start : 0;
30760        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30761
30762        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30763        this.field.dom.value = ap;
30764        this.first.setDisabled(ap == 1);
30765        this.prev.setDisabled(ap == 1);
30766        this.next.setDisabled(ap == ps);
30767        this.last.setDisabled(ap == ps);
30768        this.loading.enable();
30769        this.updateInfo();
30770     },
30771
30772     // private
30773     getPageData : function(){
30774         var total = this.ds.getTotalCount();
30775         return {
30776             total : total,
30777             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30778             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30779         };
30780     },
30781
30782     // private
30783     onLoadError : function(){
30784         this.loading.enable();
30785     },
30786
30787     // private
30788     onPagingKeydown : function(e){
30789         var k = e.getKey();
30790         var d = this.getPageData();
30791         if(k == e.RETURN){
30792             var v = this.field.dom.value, pageNum;
30793             if(!v || isNaN(pageNum = parseInt(v, 10))){
30794                 this.field.dom.value = d.activePage;
30795                 return;
30796             }
30797             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30798             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30799             e.stopEvent();
30800         }
30801         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))
30802         {
30803           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30804           this.field.dom.value = pageNum;
30805           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30806           e.stopEvent();
30807         }
30808         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30809         {
30810           var v = this.field.dom.value, pageNum; 
30811           var increment = (e.shiftKey) ? 10 : 1;
30812           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30813             increment *= -1;
30814           }
30815           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30816             this.field.dom.value = d.activePage;
30817             return;
30818           }
30819           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30820           {
30821             this.field.dom.value = parseInt(v, 10) + increment;
30822             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30823             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30824           }
30825           e.stopEvent();
30826         }
30827     },
30828
30829     // private
30830     beforeLoad : function(){
30831         if(this.loading){
30832             this.loading.disable();
30833         }
30834     },
30835
30836     // private
30837     onClick : function(which){
30838         var ds = this.ds;
30839         switch(which){
30840             case "first":
30841                 ds.load({params:{start: 0, limit: this.pageSize}});
30842             break;
30843             case "prev":
30844                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30845             break;
30846             case "next":
30847                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30848             break;
30849             case "last":
30850                 var total = ds.getTotalCount();
30851                 var extra = total % this.pageSize;
30852                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30853                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30854             break;
30855             case "refresh":
30856                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30857             break;
30858         }
30859     },
30860
30861     /**
30862      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30863      * @param {Roo.data.Store} store The data store to unbind
30864      */
30865     unbind : function(ds){
30866         ds.un("beforeload", this.beforeLoad, this);
30867         ds.un("load", this.onLoad, this);
30868         ds.un("loadexception", this.onLoadError, this);
30869         ds.un("remove", this.updateInfo, this);
30870         ds.un("add", this.updateInfo, this);
30871         this.ds = undefined;
30872     },
30873
30874     /**
30875      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30876      * @param {Roo.data.Store} store The data store to bind
30877      */
30878     bind : function(ds){
30879         ds.on("beforeload", this.beforeLoad, this);
30880         ds.on("load", this.onLoad, this);
30881         ds.on("loadexception", this.onLoadError, this);
30882         ds.on("remove", this.updateInfo, this);
30883         ds.on("add", this.updateInfo, this);
30884         this.ds = ds;
30885     }
30886 });/*
30887  * Based on:
30888  * Ext JS Library 1.1.1
30889  * Copyright(c) 2006-2007, Ext JS, LLC.
30890  *
30891  * Originally Released Under LGPL - original licence link has changed is not relivant.
30892  *
30893  * Fork - LGPL
30894  * <script type="text/javascript">
30895  */
30896
30897 /**
30898  * @class Roo.Resizable
30899  * @extends Roo.util.Observable
30900  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30901  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30902  * 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
30903  * the element will be wrapped for you automatically.</p>
30904  * <p>Here is the list of valid resize handles:</p>
30905  * <pre>
30906 Value   Description
30907 ------  -------------------
30908  'n'     north
30909  's'     south
30910  'e'     east
30911  'w'     west
30912  'nw'    northwest
30913  'sw'    southwest
30914  'se'    southeast
30915  'ne'    northeast
30916  'hd'    horizontal drag
30917  'all'   all
30918 </pre>
30919  * <p>Here's an example showing the creation of a typical Resizable:</p>
30920  * <pre><code>
30921 var resizer = new Roo.Resizable("element-id", {
30922     handles: 'all',
30923     minWidth: 200,
30924     minHeight: 100,
30925     maxWidth: 500,
30926     maxHeight: 400,
30927     pinned: true
30928 });
30929 resizer.on("resize", myHandler);
30930 </code></pre>
30931  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30932  * resizer.east.setDisplayed(false);</p>
30933  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30934  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30935  * resize operation's new size (defaults to [0, 0])
30936  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30937  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30938  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30939  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30940  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30941  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30942  * @cfg {Number} width The width of the element in pixels (defaults to null)
30943  * @cfg {Number} height The height of the element in pixels (defaults to null)
30944  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30945  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30946  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30947  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30948  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30949  * in favor of the handles config option (defaults to false)
30950  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30951  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30952  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30953  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30954  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30955  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30956  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30957  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30958  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30959  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30960  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30961  * @constructor
30962  * Create a new resizable component
30963  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30964  * @param {Object} config configuration options
30965   */
30966 Roo.Resizable = function(el, config)
30967 {
30968     this.el = Roo.get(el);
30969
30970     if(config && config.wrap){
30971         config.resizeChild = this.el;
30972         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30973         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30974         this.el.setStyle("overflow", "hidden");
30975         this.el.setPositioning(config.resizeChild.getPositioning());
30976         config.resizeChild.clearPositioning();
30977         if(!config.width || !config.height){
30978             var csize = config.resizeChild.getSize();
30979             this.el.setSize(csize.width, csize.height);
30980         }
30981         if(config.pinned && !config.adjustments){
30982             config.adjustments = "auto";
30983         }
30984     }
30985
30986     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30987     this.proxy.unselectable();
30988     this.proxy.enableDisplayMode('block');
30989
30990     Roo.apply(this, config);
30991
30992     if(this.pinned){
30993         this.disableTrackOver = true;
30994         this.el.addClass("x-resizable-pinned");
30995     }
30996     // if the element isn't positioned, make it relative
30997     var position = this.el.getStyle("position");
30998     if(position != "absolute" && position != "fixed"){
30999         this.el.setStyle("position", "relative");
31000     }
31001     if(!this.handles){ // no handles passed, must be legacy style
31002         this.handles = 's,e,se';
31003         if(this.multiDirectional){
31004             this.handles += ',n,w';
31005         }
31006     }
31007     if(this.handles == "all"){
31008         this.handles = "n s e w ne nw se sw";
31009     }
31010     var hs = this.handles.split(/\s*?[,;]\s*?| /);
31011     var ps = Roo.Resizable.positions;
31012     for(var i = 0, len = hs.length; i < len; i++){
31013         if(hs[i] && ps[hs[i]]){
31014             var pos = ps[hs[i]];
31015             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
31016         }
31017     }
31018     // legacy
31019     this.corner = this.southeast;
31020     
31021     // updateBox = the box can move..
31022     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
31023         this.updateBox = true;
31024     }
31025
31026     this.activeHandle = null;
31027
31028     if(this.resizeChild){
31029         if(typeof this.resizeChild == "boolean"){
31030             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
31031         }else{
31032             this.resizeChild = Roo.get(this.resizeChild, true);
31033         }
31034     }
31035     
31036     if(this.adjustments == "auto"){
31037         var rc = this.resizeChild;
31038         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
31039         if(rc && (hw || hn)){
31040             rc.position("relative");
31041             rc.setLeft(hw ? hw.el.getWidth() : 0);
31042             rc.setTop(hn ? hn.el.getHeight() : 0);
31043         }
31044         this.adjustments = [
31045             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31046             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31047         ];
31048     }
31049
31050     if(this.draggable){
31051         this.dd = this.dynamic ?
31052             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31053         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31054     }
31055
31056     // public events
31057     this.addEvents({
31058         /**
31059          * @event beforeresize
31060          * Fired before resize is allowed. Set enabled to false to cancel resize.
31061          * @param {Roo.Resizable} this
31062          * @param {Roo.EventObject} e The mousedown event
31063          */
31064         "beforeresize" : true,
31065         /**
31066          * @event resizing
31067          * Fired a resizing.
31068          * @param {Roo.Resizable} this
31069          * @param {Number} x The new x position
31070          * @param {Number} y The new y position
31071          * @param {Number} w The new w width
31072          * @param {Number} h The new h hight
31073          * @param {Roo.EventObject} e The mouseup event
31074          */
31075         "resizing" : true,
31076         /**
31077          * @event resize
31078          * Fired after a resize.
31079          * @param {Roo.Resizable} this
31080          * @param {Number} width The new width
31081          * @param {Number} height The new height
31082          * @param {Roo.EventObject} e The mouseup event
31083          */
31084         "resize" : true
31085     });
31086
31087     if(this.width !== null && this.height !== null){
31088         this.resizeTo(this.width, this.height);
31089     }else{
31090         this.updateChildSize();
31091     }
31092     if(Roo.isIE){
31093         this.el.dom.style.zoom = 1;
31094     }
31095     Roo.Resizable.superclass.constructor.call(this);
31096 };
31097
31098 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31099         resizeChild : false,
31100         adjustments : [0, 0],
31101         minWidth : 5,
31102         minHeight : 5,
31103         maxWidth : 10000,
31104         maxHeight : 10000,
31105         enabled : true,
31106         animate : false,
31107         duration : .35,
31108         dynamic : false,
31109         handles : false,
31110         multiDirectional : false,
31111         disableTrackOver : false,
31112         easing : 'easeOutStrong',
31113         widthIncrement : 0,
31114         heightIncrement : 0,
31115         pinned : false,
31116         width : null,
31117         height : null,
31118         preserveRatio : false,
31119         transparent: false,
31120         minX: 0,
31121         minY: 0,
31122         draggable: false,
31123
31124         /**
31125          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31126          */
31127         constrainTo: undefined,
31128         /**
31129          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31130          */
31131         resizeRegion: undefined,
31132
31133
31134     /**
31135      * Perform a manual resize
31136      * @param {Number} width
31137      * @param {Number} height
31138      */
31139     resizeTo : function(width, height){
31140         this.el.setSize(width, height);
31141         this.updateChildSize();
31142         this.fireEvent("resize", this, width, height, null);
31143     },
31144
31145     // private
31146     startSizing : function(e, handle){
31147         this.fireEvent("beforeresize", this, e);
31148         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31149
31150             if(!this.overlay){
31151                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31152                 this.overlay.unselectable();
31153                 this.overlay.enableDisplayMode("block");
31154                 this.overlay.on("mousemove", this.onMouseMove, this);
31155                 this.overlay.on("mouseup", this.onMouseUp, this);
31156             }
31157             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31158
31159             this.resizing = true;
31160             this.startBox = this.el.getBox();
31161             this.startPoint = e.getXY();
31162             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31163                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31164
31165             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31166             this.overlay.show();
31167
31168             if(this.constrainTo) {
31169                 var ct = Roo.get(this.constrainTo);
31170                 this.resizeRegion = ct.getRegion().adjust(
31171                     ct.getFrameWidth('t'),
31172                     ct.getFrameWidth('l'),
31173                     -ct.getFrameWidth('b'),
31174                     -ct.getFrameWidth('r')
31175                 );
31176             }
31177
31178             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31179             this.proxy.show();
31180             this.proxy.setBox(this.startBox);
31181             if(!this.dynamic){
31182                 this.proxy.setStyle('visibility', 'visible');
31183             }
31184         }
31185     },
31186
31187     // private
31188     onMouseDown : function(handle, e){
31189         if(this.enabled){
31190             e.stopEvent();
31191             this.activeHandle = handle;
31192             this.startSizing(e, handle);
31193         }
31194     },
31195
31196     // private
31197     onMouseUp : function(e){
31198         var size = this.resizeElement();
31199         this.resizing = false;
31200         this.handleOut();
31201         this.overlay.hide();
31202         this.proxy.hide();
31203         this.fireEvent("resize", this, size.width, size.height, e);
31204     },
31205
31206     // private
31207     updateChildSize : function(){
31208         
31209         if(this.resizeChild){
31210             var el = this.el;
31211             var child = this.resizeChild;
31212             var adj = this.adjustments;
31213             if(el.dom.offsetWidth){
31214                 var b = el.getSize(true);
31215                 child.setSize(b.width+adj[0], b.height+adj[1]);
31216             }
31217             // Second call here for IE
31218             // The first call enables instant resizing and
31219             // the second call corrects scroll bars if they
31220             // exist
31221             if(Roo.isIE){
31222                 setTimeout(function(){
31223                     if(el.dom.offsetWidth){
31224                         var b = el.getSize(true);
31225                         child.setSize(b.width+adj[0], b.height+adj[1]);
31226                     }
31227                 }, 10);
31228             }
31229         }
31230     },
31231
31232     // private
31233     snap : function(value, inc, min){
31234         if(!inc || !value) {
31235             return value;
31236         }
31237         var newValue = value;
31238         var m = value % inc;
31239         if(m > 0){
31240             if(m > (inc/2)){
31241                 newValue = value + (inc-m);
31242             }else{
31243                 newValue = value - m;
31244             }
31245         }
31246         return Math.max(min, newValue);
31247     },
31248
31249     // private
31250     resizeElement : function(){
31251         var box = this.proxy.getBox();
31252         if(this.updateBox){
31253             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31254         }else{
31255             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31256         }
31257         this.updateChildSize();
31258         if(!this.dynamic){
31259             this.proxy.hide();
31260         }
31261         return box;
31262     },
31263
31264     // private
31265     constrain : function(v, diff, m, mx){
31266         if(v - diff < m){
31267             diff = v - m;
31268         }else if(v - diff > mx){
31269             diff = mx - v;
31270         }
31271         return diff;
31272     },
31273
31274     // private
31275     onMouseMove : function(e){
31276         
31277         if(this.enabled){
31278             try{// try catch so if something goes wrong the user doesn't get hung
31279
31280             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31281                 return;
31282             }
31283
31284             //var curXY = this.startPoint;
31285             var curSize = this.curSize || this.startBox;
31286             var x = this.startBox.x, y = this.startBox.y;
31287             var ox = x, oy = y;
31288             var w = curSize.width, h = curSize.height;
31289             var ow = w, oh = h;
31290             var mw = this.minWidth, mh = this.minHeight;
31291             var mxw = this.maxWidth, mxh = this.maxHeight;
31292             var wi = this.widthIncrement;
31293             var hi = this.heightIncrement;
31294
31295             var eventXY = e.getXY();
31296             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31297             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31298
31299             var pos = this.activeHandle.position;
31300
31301             switch(pos){
31302                 case "east":
31303                     w += diffX;
31304                     w = Math.min(Math.max(mw, w), mxw);
31305                     break;
31306              
31307                 case "south":
31308                     h += diffY;
31309                     h = Math.min(Math.max(mh, h), mxh);
31310                     break;
31311                 case "southeast":
31312                     w += diffX;
31313                     h += diffY;
31314                     w = Math.min(Math.max(mw, w), mxw);
31315                     h = Math.min(Math.max(mh, h), mxh);
31316                     break;
31317                 case "north":
31318                     diffY = this.constrain(h, diffY, mh, mxh);
31319                     y += diffY;
31320                     h -= diffY;
31321                     break;
31322                 case "hdrag":
31323                     
31324                     if (wi) {
31325                         var adiffX = Math.abs(diffX);
31326                         var sub = (adiffX % wi); // how much 
31327                         if (sub > (wi/2)) { // far enough to snap
31328                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31329                         } else {
31330                             // remove difference.. 
31331                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31332                         }
31333                     }
31334                     x += diffX;
31335                     x = Math.max(this.minX, x);
31336                     break;
31337                 case "west":
31338                     diffX = this.constrain(w, diffX, mw, mxw);
31339                     x += diffX;
31340                     w -= diffX;
31341                     break;
31342                 case "northeast":
31343                     w += diffX;
31344                     w = Math.min(Math.max(mw, w), mxw);
31345                     diffY = this.constrain(h, diffY, mh, mxh);
31346                     y += diffY;
31347                     h -= diffY;
31348                     break;
31349                 case "northwest":
31350                     diffX = this.constrain(w, diffX, mw, mxw);
31351                     diffY = this.constrain(h, diffY, mh, mxh);
31352                     y += diffY;
31353                     h -= diffY;
31354                     x += diffX;
31355                     w -= diffX;
31356                     break;
31357                case "southwest":
31358                     diffX = this.constrain(w, diffX, mw, mxw);
31359                     h += diffY;
31360                     h = Math.min(Math.max(mh, h), mxh);
31361                     x += diffX;
31362                     w -= diffX;
31363                     break;
31364             }
31365
31366             var sw = this.snap(w, wi, mw);
31367             var sh = this.snap(h, hi, mh);
31368             if(sw != w || sh != h){
31369                 switch(pos){
31370                     case "northeast":
31371                         y -= sh - h;
31372                     break;
31373                     case "north":
31374                         y -= sh - h;
31375                         break;
31376                     case "southwest":
31377                         x -= sw - w;
31378                     break;
31379                     case "west":
31380                         x -= sw - w;
31381                         break;
31382                     case "northwest":
31383                         x -= sw - w;
31384                         y -= sh - h;
31385                     break;
31386                 }
31387                 w = sw;
31388                 h = sh;
31389             }
31390
31391             if(this.preserveRatio){
31392                 switch(pos){
31393                     case "southeast":
31394                     case "east":
31395                         h = oh * (w/ow);
31396                         h = Math.min(Math.max(mh, h), mxh);
31397                         w = ow * (h/oh);
31398                        break;
31399                     case "south":
31400                         w = ow * (h/oh);
31401                         w = Math.min(Math.max(mw, w), mxw);
31402                         h = oh * (w/ow);
31403                         break;
31404                     case "northeast":
31405                         w = ow * (h/oh);
31406                         w = Math.min(Math.max(mw, w), mxw);
31407                         h = oh * (w/ow);
31408                     break;
31409                     case "north":
31410                         var tw = w;
31411                         w = ow * (h/oh);
31412                         w = Math.min(Math.max(mw, w), mxw);
31413                         h = oh * (w/ow);
31414                         x += (tw - w) / 2;
31415                         break;
31416                     case "southwest":
31417                         h = oh * (w/ow);
31418                         h = Math.min(Math.max(mh, h), mxh);
31419                         var tw = w;
31420                         w = ow * (h/oh);
31421                         x += tw - w;
31422                         break;
31423                     case "west":
31424                         var th = h;
31425                         h = oh * (w/ow);
31426                         h = Math.min(Math.max(mh, h), mxh);
31427                         y += (th - h) / 2;
31428                         var tw = w;
31429                         w = ow * (h/oh);
31430                         x += tw - w;
31431                        break;
31432                     case "northwest":
31433                         var tw = w;
31434                         var th = h;
31435                         h = oh * (w/ow);
31436                         h = Math.min(Math.max(mh, h), mxh);
31437                         w = ow * (h/oh);
31438                         y += th - h;
31439                         x += tw - w;
31440                        break;
31441
31442                 }
31443             }
31444             if (pos == 'hdrag') {
31445                 w = ow;
31446             }
31447             this.proxy.setBounds(x, y, w, h);
31448             if(this.dynamic){
31449                 this.resizeElement();
31450             }
31451             }catch(e){}
31452         }
31453         this.fireEvent("resizing", this, x, y, w, h, e);
31454     },
31455
31456     // private
31457     handleOver : function(){
31458         if(this.enabled){
31459             this.el.addClass("x-resizable-over");
31460         }
31461     },
31462
31463     // private
31464     handleOut : function(){
31465         if(!this.resizing){
31466             this.el.removeClass("x-resizable-over");
31467         }
31468     },
31469
31470     /**
31471      * Returns the element this component is bound to.
31472      * @return {Roo.Element}
31473      */
31474     getEl : function(){
31475         return this.el;
31476     },
31477
31478     /**
31479      * Returns the resizeChild element (or null).
31480      * @return {Roo.Element}
31481      */
31482     getResizeChild : function(){
31483         return this.resizeChild;
31484     },
31485     groupHandler : function()
31486     {
31487         
31488     },
31489     /**
31490      * Destroys this resizable. If the element was wrapped and
31491      * removeEl is not true then the element remains.
31492      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31493      */
31494     destroy : function(removeEl){
31495         this.proxy.remove();
31496         if(this.overlay){
31497             this.overlay.removeAllListeners();
31498             this.overlay.remove();
31499         }
31500         var ps = Roo.Resizable.positions;
31501         for(var k in ps){
31502             if(typeof ps[k] != "function" && this[ps[k]]){
31503                 var h = this[ps[k]];
31504                 h.el.removeAllListeners();
31505                 h.el.remove();
31506             }
31507         }
31508         if(removeEl){
31509             this.el.update("");
31510             this.el.remove();
31511         }
31512     }
31513 });
31514
31515 // private
31516 // hash to map config positions to true positions
31517 Roo.Resizable.positions = {
31518     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31519     hd: "hdrag"
31520 };
31521
31522 // private
31523 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31524     if(!this.tpl){
31525         // only initialize the template if resizable is used
31526         var tpl = Roo.DomHelper.createTemplate(
31527             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31528         );
31529         tpl.compile();
31530         Roo.Resizable.Handle.prototype.tpl = tpl;
31531     }
31532     this.position = pos;
31533     this.rz = rz;
31534     // show north drag fro topdra
31535     var handlepos = pos == 'hdrag' ? 'north' : pos;
31536     
31537     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31538     if (pos == 'hdrag') {
31539         this.el.setStyle('cursor', 'pointer');
31540     }
31541     this.el.unselectable();
31542     if(transparent){
31543         this.el.setOpacity(0);
31544     }
31545     this.el.on("mousedown", this.onMouseDown, this);
31546     if(!disableTrackOver){
31547         this.el.on("mouseover", this.onMouseOver, this);
31548         this.el.on("mouseout", this.onMouseOut, this);
31549     }
31550 };
31551
31552 // private
31553 Roo.Resizable.Handle.prototype = {
31554     afterResize : function(rz){
31555         Roo.log('after?');
31556         // do nothing
31557     },
31558     // private
31559     onMouseDown : function(e){
31560         this.rz.onMouseDown(this, e);
31561     },
31562     // private
31563     onMouseOver : function(e){
31564         this.rz.handleOver(this, e);
31565     },
31566     // private
31567     onMouseOut : function(e){
31568         this.rz.handleOut(this, e);
31569     }
31570 };/*
31571  * Based on:
31572  * Ext JS Library 1.1.1
31573  * Copyright(c) 2006-2007, Ext JS, LLC.
31574  *
31575  * Originally Released Under LGPL - original licence link has changed is not relivant.
31576  *
31577  * Fork - LGPL
31578  * <script type="text/javascript">
31579  */
31580
31581 /**
31582  * @class Roo.Editor
31583  * @extends Roo.Component
31584  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31585  * @constructor
31586  * Create a new Editor
31587  * @param {Roo.form.Field} field The Field object (or descendant)
31588  * @param {Object} config The config object
31589  */
31590 Roo.Editor = function(field, config){
31591     Roo.Editor.superclass.constructor.call(this, config);
31592     this.field = field;
31593     this.addEvents({
31594         /**
31595              * @event beforestartedit
31596              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31597              * false from the handler of this event.
31598              * @param {Editor} this
31599              * @param {Roo.Element} boundEl The underlying element bound to this editor
31600              * @param {Mixed} value The field value being set
31601              */
31602         "beforestartedit" : true,
31603         /**
31604              * @event startedit
31605              * Fires when this editor is displayed
31606              * @param {Roo.Element} boundEl The underlying element bound to this editor
31607              * @param {Mixed} value The starting field value
31608              */
31609         "startedit" : true,
31610         /**
31611              * @event beforecomplete
31612              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31613              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31614              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31615              * event will not fire since no edit actually occurred.
31616              * @param {Editor} this
31617              * @param {Mixed} value The current field value
31618              * @param {Mixed} startValue The original field value
31619              */
31620         "beforecomplete" : true,
31621         /**
31622              * @event complete
31623              * Fires after editing is complete and any changed value has been written to the underlying field.
31624              * @param {Editor} this
31625              * @param {Mixed} value The current field value
31626              * @param {Mixed} startValue The original field value
31627              */
31628         "complete" : true,
31629         /**
31630          * @event specialkey
31631          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31632          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31633          * @param {Roo.form.Field} this
31634          * @param {Roo.EventObject} e The event object
31635          */
31636         "specialkey" : true
31637     });
31638 };
31639
31640 Roo.extend(Roo.Editor, Roo.Component, {
31641     /**
31642      * @cfg {Boolean/String} autosize
31643      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31644      * or "height" to adopt the height only (defaults to false)
31645      */
31646     /**
31647      * @cfg {Boolean} revertInvalid
31648      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31649      * validation fails (defaults to true)
31650      */
31651     /**
31652      * @cfg {Boolean} ignoreNoChange
31653      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31654      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31655      * will never be ignored.
31656      */
31657     /**
31658      * @cfg {Boolean} hideEl
31659      * False to keep the bound element visible while the editor is displayed (defaults to true)
31660      */
31661     /**
31662      * @cfg {Mixed} value
31663      * The data value of the underlying field (defaults to "")
31664      */
31665     value : "",
31666     /**
31667      * @cfg {String} alignment
31668      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31669      */
31670     alignment: "c-c?",
31671     /**
31672      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31673      * for bottom-right shadow (defaults to "frame")
31674      */
31675     shadow : "frame",
31676     /**
31677      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31678      */
31679     constrain : false,
31680     /**
31681      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31682      */
31683     completeOnEnter : false,
31684     /**
31685      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31686      */
31687     cancelOnEsc : false,
31688     /**
31689      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31690      */
31691     updateEl : false,
31692
31693     // private
31694     onRender : function(ct, position){
31695         this.el = new Roo.Layer({
31696             shadow: this.shadow,
31697             cls: "x-editor",
31698             parentEl : ct,
31699             shim : this.shim,
31700             shadowOffset:4,
31701             id: this.id,
31702             constrain: this.constrain
31703         });
31704         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31705         if(this.field.msgTarget != 'title'){
31706             this.field.msgTarget = 'qtip';
31707         }
31708         this.field.render(this.el);
31709         if(Roo.isGecko){
31710             this.field.el.dom.setAttribute('autocomplete', 'off');
31711         }
31712         this.field.on("specialkey", this.onSpecialKey, this);
31713         if(this.swallowKeys){
31714             this.field.el.swallowEvent(['keydown','keypress']);
31715         }
31716         this.field.show();
31717         this.field.on("blur", this.onBlur, this);
31718         if(this.field.grow){
31719             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31720         }
31721     },
31722
31723     onSpecialKey : function(field, e)
31724     {
31725         //Roo.log('editor onSpecialKey');
31726         if(this.completeOnEnter && e.getKey() == e.ENTER){
31727             e.stopEvent();
31728             this.completeEdit();
31729             return;
31730         }
31731         // do not fire special key otherwise it might hide close the editor...
31732         if(e.getKey() == e.ENTER){    
31733             return;
31734         }
31735         if(this.cancelOnEsc && e.getKey() == e.ESC){
31736             this.cancelEdit();
31737             return;
31738         } 
31739         this.fireEvent('specialkey', field, e);
31740     
31741     },
31742
31743     /**
31744      * Starts the editing process and shows the editor.
31745      * @param {String/HTMLElement/Element} el The element to edit
31746      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31747       * to the innerHTML of el.
31748      */
31749     startEdit : function(el, value){
31750         if(this.editing){
31751             this.completeEdit();
31752         }
31753         this.boundEl = Roo.get(el);
31754         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31755         if(!this.rendered){
31756             this.render(this.parentEl || document.body);
31757         }
31758         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31759             return;
31760         }
31761         this.startValue = v;
31762         this.field.setValue(v);
31763         if(this.autoSize){
31764             var sz = this.boundEl.getSize();
31765             switch(this.autoSize){
31766                 case "width":
31767                 this.setSize(sz.width,  "");
31768                 break;
31769                 case "height":
31770                 this.setSize("",  sz.height);
31771                 break;
31772                 default:
31773                 this.setSize(sz.width,  sz.height);
31774             }
31775         }
31776         this.el.alignTo(this.boundEl, this.alignment);
31777         this.editing = true;
31778         if(Roo.QuickTips){
31779             Roo.QuickTips.disable();
31780         }
31781         this.show();
31782     },
31783
31784     /**
31785      * Sets the height and width of this editor.
31786      * @param {Number} width The new width
31787      * @param {Number} height The new height
31788      */
31789     setSize : function(w, h){
31790         this.field.setSize(w, h);
31791         if(this.el){
31792             this.el.sync();
31793         }
31794     },
31795
31796     /**
31797      * Realigns the editor to the bound field based on the current alignment config value.
31798      */
31799     realign : function(){
31800         this.el.alignTo(this.boundEl, this.alignment);
31801     },
31802
31803     /**
31804      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31805      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31806      */
31807     completeEdit : function(remainVisible){
31808         if(!this.editing){
31809             return;
31810         }
31811         var v = this.getValue();
31812         if(this.revertInvalid !== false && !this.field.isValid()){
31813             v = this.startValue;
31814             this.cancelEdit(true);
31815         }
31816         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31817             this.editing = false;
31818             this.hide();
31819             return;
31820         }
31821         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31822             this.editing = false;
31823             if(this.updateEl && this.boundEl){
31824                 this.boundEl.update(v);
31825             }
31826             if(remainVisible !== true){
31827                 this.hide();
31828             }
31829             this.fireEvent("complete", this, v, this.startValue);
31830         }
31831     },
31832
31833     // private
31834     onShow : function(){
31835         this.el.show();
31836         if(this.hideEl !== false){
31837             this.boundEl.hide();
31838         }
31839         this.field.show();
31840         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31841             this.fixIEFocus = true;
31842             this.deferredFocus.defer(50, this);
31843         }else{
31844             this.field.focus();
31845         }
31846         this.fireEvent("startedit", this.boundEl, this.startValue);
31847     },
31848
31849     deferredFocus : function(){
31850         if(this.editing){
31851             this.field.focus();
31852         }
31853     },
31854
31855     /**
31856      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31857      * reverted to the original starting value.
31858      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31859      * cancel (defaults to false)
31860      */
31861     cancelEdit : function(remainVisible){
31862         if(this.editing){
31863             this.setValue(this.startValue);
31864             if(remainVisible !== true){
31865                 this.hide();
31866             }
31867         }
31868     },
31869
31870     // private
31871     onBlur : function(){
31872         if(this.allowBlur !== true && this.editing){
31873             this.completeEdit();
31874         }
31875     },
31876
31877     // private
31878     onHide : function(){
31879         if(this.editing){
31880             this.completeEdit();
31881             return;
31882         }
31883         this.field.blur();
31884         if(this.field.collapse){
31885             this.field.collapse();
31886         }
31887         this.el.hide();
31888         if(this.hideEl !== false){
31889             this.boundEl.show();
31890         }
31891         if(Roo.QuickTips){
31892             Roo.QuickTips.enable();
31893         }
31894     },
31895
31896     /**
31897      * Sets the data value of the editor
31898      * @param {Mixed} value Any valid value supported by the underlying field
31899      */
31900     setValue : function(v){
31901         this.field.setValue(v);
31902     },
31903
31904     /**
31905      * Gets the data value of the editor
31906      * @return {Mixed} The data value
31907      */
31908     getValue : function(){
31909         return this.field.getValue();
31910     }
31911 });/*
31912  * Based on:
31913  * Ext JS Library 1.1.1
31914  * Copyright(c) 2006-2007, Ext JS, LLC.
31915  *
31916  * Originally Released Under LGPL - original licence link has changed is not relivant.
31917  *
31918  * Fork - LGPL
31919  * <script type="text/javascript">
31920  */
31921  
31922 /**
31923  * @class Roo.BasicDialog
31924  * @extends Roo.util.Observable
31925  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31926  * <pre><code>
31927 var dlg = new Roo.BasicDialog("my-dlg", {
31928     height: 200,
31929     width: 300,
31930     minHeight: 100,
31931     minWidth: 150,
31932     modal: true,
31933     proxyDrag: true,
31934     shadow: true
31935 });
31936 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31937 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31938 dlg.addButton('Cancel', dlg.hide, dlg);
31939 dlg.show();
31940 </code></pre>
31941   <b>A Dialog should always be a direct child of the body element.</b>
31942  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31943  * @cfg {String} title Default text to display in the title bar (defaults to null)
31944  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31945  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31946  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31947  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31948  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31949  * (defaults to null with no animation)
31950  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31951  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31952  * property for valid values (defaults to 'all')
31953  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31954  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31955  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31956  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31957  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31958  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31959  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31960  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31961  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31962  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31963  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31964  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31965  * draggable = true (defaults to false)
31966  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31967  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31968  * shadow (defaults to false)
31969  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31970  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31971  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31972  * @cfg {Array} buttons Array of buttons
31973  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31974  * @constructor
31975  * Create a new BasicDialog.
31976  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31977  * @param {Object} config Configuration options
31978  */
31979 Roo.BasicDialog = function(el, config){
31980     this.el = Roo.get(el);
31981     var dh = Roo.DomHelper;
31982     if(!this.el && config && config.autoCreate){
31983         if(typeof config.autoCreate == "object"){
31984             if(!config.autoCreate.id){
31985                 config.autoCreate.id = el;
31986             }
31987             this.el = dh.append(document.body,
31988                         config.autoCreate, true);
31989         }else{
31990             this.el = dh.append(document.body,
31991                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31992         }
31993     }
31994     el = this.el;
31995     el.setDisplayed(true);
31996     el.hide = this.hideAction;
31997     this.id = el.id;
31998     el.addClass("x-dlg");
31999
32000     Roo.apply(this, config);
32001
32002     this.proxy = el.createProxy("x-dlg-proxy");
32003     this.proxy.hide = this.hideAction;
32004     this.proxy.setOpacity(.5);
32005     this.proxy.hide();
32006
32007     if(config.width){
32008         el.setWidth(config.width);
32009     }
32010     if(config.height){
32011         el.setHeight(config.height);
32012     }
32013     this.size = el.getSize();
32014     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
32015         this.xy = [config.x,config.y];
32016     }else{
32017         this.xy = el.getCenterXY(true);
32018     }
32019     /** The header element @type Roo.Element */
32020     this.header = el.child("> .x-dlg-hd");
32021     /** The body element @type Roo.Element */
32022     this.body = el.child("> .x-dlg-bd");
32023     /** The footer element @type Roo.Element */
32024     this.footer = el.child("> .x-dlg-ft");
32025
32026     if(!this.header){
32027         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
32028     }
32029     if(!this.body){
32030         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
32031     }
32032
32033     this.header.unselectable();
32034     if(this.title){
32035         this.header.update(this.title);
32036     }
32037     // this element allows the dialog to be focused for keyboard event
32038     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
32039     this.focusEl.swallowEvent("click", true);
32040
32041     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
32042
32043     // wrap the body and footer for special rendering
32044     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32045     if(this.footer){
32046         this.bwrap.dom.appendChild(this.footer.dom);
32047     }
32048
32049     this.bg = this.el.createChild({
32050         tag: "div", cls:"x-dlg-bg",
32051         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32052     });
32053     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32054
32055
32056     if(this.autoScroll !== false && !this.autoTabs){
32057         this.body.setStyle("overflow", "auto");
32058     }
32059
32060     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32061
32062     if(this.closable !== false){
32063         this.el.addClass("x-dlg-closable");
32064         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32065         this.close.on("click", this.closeClick, this);
32066         this.close.addClassOnOver("x-dlg-close-over");
32067     }
32068     if(this.collapsible !== false){
32069         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32070         this.collapseBtn.on("click", this.collapseClick, this);
32071         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32072         this.header.on("dblclick", this.collapseClick, this);
32073     }
32074     if(this.resizable !== false){
32075         this.el.addClass("x-dlg-resizable");
32076         this.resizer = new Roo.Resizable(el, {
32077             minWidth: this.minWidth || 80,
32078             minHeight:this.minHeight || 80,
32079             handles: this.resizeHandles || "all",
32080             pinned: true
32081         });
32082         this.resizer.on("beforeresize", this.beforeResize, this);
32083         this.resizer.on("resize", this.onResize, this);
32084     }
32085     if(this.draggable !== false){
32086         el.addClass("x-dlg-draggable");
32087         if (!this.proxyDrag) {
32088             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32089         }
32090         else {
32091             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32092         }
32093         dd.setHandleElId(this.header.id);
32094         dd.endDrag = this.endMove.createDelegate(this);
32095         dd.startDrag = this.startMove.createDelegate(this);
32096         dd.onDrag = this.onDrag.createDelegate(this);
32097         dd.scroll = false;
32098         this.dd = dd;
32099     }
32100     if(this.modal){
32101         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32102         this.mask.enableDisplayMode("block");
32103         this.mask.hide();
32104         this.el.addClass("x-dlg-modal");
32105     }
32106     if(this.shadow){
32107         this.shadow = new Roo.Shadow({
32108             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32109             offset : this.shadowOffset
32110         });
32111     }else{
32112         this.shadowOffset = 0;
32113     }
32114     if(Roo.useShims && this.shim !== false){
32115         this.shim = this.el.createShim();
32116         this.shim.hide = this.hideAction;
32117         this.shim.hide();
32118     }else{
32119         this.shim = false;
32120     }
32121     if(this.autoTabs){
32122         this.initTabs();
32123     }
32124     if (this.buttons) { 
32125         var bts= this.buttons;
32126         this.buttons = [];
32127         Roo.each(bts, function(b) {
32128             this.addButton(b);
32129         }, this);
32130     }
32131     
32132     
32133     this.addEvents({
32134         /**
32135          * @event keydown
32136          * Fires when a key is pressed
32137          * @param {Roo.BasicDialog} this
32138          * @param {Roo.EventObject} e
32139          */
32140         "keydown" : true,
32141         /**
32142          * @event move
32143          * Fires when this dialog is moved by the user.
32144          * @param {Roo.BasicDialog} this
32145          * @param {Number} x The new page X
32146          * @param {Number} y The new page Y
32147          */
32148         "move" : true,
32149         /**
32150          * @event resize
32151          * Fires when this dialog is resized by the user.
32152          * @param {Roo.BasicDialog} this
32153          * @param {Number} width The new width
32154          * @param {Number} height The new height
32155          */
32156         "resize" : true,
32157         /**
32158          * @event beforehide
32159          * Fires before this dialog is hidden.
32160          * @param {Roo.BasicDialog} this
32161          */
32162         "beforehide" : true,
32163         /**
32164          * @event hide
32165          * Fires when this dialog is hidden.
32166          * @param {Roo.BasicDialog} this
32167          */
32168         "hide" : true,
32169         /**
32170          * @event beforeshow
32171          * Fires before this dialog is shown.
32172          * @param {Roo.BasicDialog} this
32173          */
32174         "beforeshow" : true,
32175         /**
32176          * @event show
32177          * Fires when this dialog is shown.
32178          * @param {Roo.BasicDialog} this
32179          */
32180         "show" : true
32181     });
32182     el.on("keydown", this.onKeyDown, this);
32183     el.on("mousedown", this.toFront, this);
32184     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32185     this.el.hide();
32186     Roo.DialogManager.register(this);
32187     Roo.BasicDialog.superclass.constructor.call(this);
32188 };
32189
32190 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32191     shadowOffset: Roo.isIE ? 6 : 5,
32192     minHeight: 80,
32193     minWidth: 200,
32194     minButtonWidth: 75,
32195     defaultButton: null,
32196     buttonAlign: "right",
32197     tabTag: 'div',
32198     firstShow: true,
32199
32200     /**
32201      * Sets the dialog title text
32202      * @param {String} text The title text to display
32203      * @return {Roo.BasicDialog} this
32204      */
32205     setTitle : function(text){
32206         this.header.update(text);
32207         return this;
32208     },
32209
32210     // private
32211     closeClick : function(){
32212         this.hide();
32213     },
32214
32215     // private
32216     collapseClick : function(){
32217         this[this.collapsed ? "expand" : "collapse"]();
32218     },
32219
32220     /**
32221      * Collapses the dialog to its minimized state (only the title bar is visible).
32222      * Equivalent to the user clicking the collapse dialog button.
32223      */
32224     collapse : function(){
32225         if(!this.collapsed){
32226             this.collapsed = true;
32227             this.el.addClass("x-dlg-collapsed");
32228             this.restoreHeight = this.el.getHeight();
32229             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32230         }
32231     },
32232
32233     /**
32234      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32235      * clicking the expand dialog button.
32236      */
32237     expand : function(){
32238         if(this.collapsed){
32239             this.collapsed = false;
32240             this.el.removeClass("x-dlg-collapsed");
32241             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32242         }
32243     },
32244
32245     /**
32246      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32247      * @return {Roo.TabPanel} The tabs component
32248      */
32249     initTabs : function(){
32250         var tabs = this.getTabs();
32251         while(tabs.getTab(0)){
32252             tabs.removeTab(0);
32253         }
32254         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32255             var dom = el.dom;
32256             tabs.addTab(Roo.id(dom), dom.title);
32257             dom.title = "";
32258         });
32259         tabs.activate(0);
32260         return tabs;
32261     },
32262
32263     // private
32264     beforeResize : function(){
32265         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32266     },
32267
32268     // private
32269     onResize : function(){
32270         this.refreshSize();
32271         this.syncBodyHeight();
32272         this.adjustAssets();
32273         this.focus();
32274         this.fireEvent("resize", this, this.size.width, this.size.height);
32275     },
32276
32277     // private
32278     onKeyDown : function(e){
32279         if(this.isVisible()){
32280             this.fireEvent("keydown", this, e);
32281         }
32282     },
32283
32284     /**
32285      * Resizes the dialog.
32286      * @param {Number} width
32287      * @param {Number} height
32288      * @return {Roo.BasicDialog} this
32289      */
32290     resizeTo : function(width, height){
32291         this.el.setSize(width, height);
32292         this.size = {width: width, height: height};
32293         this.syncBodyHeight();
32294         if(this.fixedcenter){
32295             this.center();
32296         }
32297         if(this.isVisible()){
32298             this.constrainXY();
32299             this.adjustAssets();
32300         }
32301         this.fireEvent("resize", this, width, height);
32302         return this;
32303     },
32304
32305
32306     /**
32307      * Resizes the dialog to fit the specified content size.
32308      * @param {Number} width
32309      * @param {Number} height
32310      * @return {Roo.BasicDialog} this
32311      */
32312     setContentSize : function(w, h){
32313         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32314         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32315         //if(!this.el.isBorderBox()){
32316             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32317             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32318         //}
32319         if(this.tabs){
32320             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32321             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32322         }
32323         this.resizeTo(w, h);
32324         return this;
32325     },
32326
32327     /**
32328      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32329      * executed in response to a particular key being pressed while the dialog is active.
32330      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32331      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32332      * @param {Function} fn The function to call
32333      * @param {Object} scope (optional) The scope of the function
32334      * @return {Roo.BasicDialog} this
32335      */
32336     addKeyListener : function(key, fn, scope){
32337         var keyCode, shift, ctrl, alt;
32338         if(typeof key == "object" && !(key instanceof Array)){
32339             keyCode = key["key"];
32340             shift = key["shift"];
32341             ctrl = key["ctrl"];
32342             alt = key["alt"];
32343         }else{
32344             keyCode = key;
32345         }
32346         var handler = function(dlg, e){
32347             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32348                 var k = e.getKey();
32349                 if(keyCode instanceof Array){
32350                     for(var i = 0, len = keyCode.length; i < len; i++){
32351                         if(keyCode[i] == k){
32352                           fn.call(scope || window, dlg, k, e);
32353                           return;
32354                         }
32355                     }
32356                 }else{
32357                     if(k == keyCode){
32358                         fn.call(scope || window, dlg, k, e);
32359                     }
32360                 }
32361             }
32362         };
32363         this.on("keydown", handler);
32364         return this;
32365     },
32366
32367     /**
32368      * Returns the TabPanel component (creates it if it doesn't exist).
32369      * Note: If you wish to simply check for the existence of tabs without creating them,
32370      * check for a null 'tabs' property.
32371      * @return {Roo.TabPanel} The tabs component
32372      */
32373     getTabs : function(){
32374         if(!this.tabs){
32375             this.el.addClass("x-dlg-auto-tabs");
32376             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32377             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32378         }
32379         return this.tabs;
32380     },
32381
32382     /**
32383      * Adds a button to the footer section of the dialog.
32384      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32385      * object or a valid Roo.DomHelper element config
32386      * @param {Function} handler The function called when the button is clicked
32387      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32388      * @return {Roo.Button} The new button
32389      */
32390     addButton : function(config, handler, scope){
32391         var dh = Roo.DomHelper;
32392         if(!this.footer){
32393             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32394         }
32395         if(!this.btnContainer){
32396             var tb = this.footer.createChild({
32397
32398                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32399                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32400             }, null, true);
32401             this.btnContainer = tb.firstChild.firstChild.firstChild;
32402         }
32403         var bconfig = {
32404             handler: handler,
32405             scope: scope,
32406             minWidth: this.minButtonWidth,
32407             hideParent:true
32408         };
32409         if(typeof config == "string"){
32410             bconfig.text = config;
32411         }else{
32412             if(config.tag){
32413                 bconfig.dhconfig = config;
32414             }else{
32415                 Roo.apply(bconfig, config);
32416             }
32417         }
32418         var fc = false;
32419         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32420             bconfig.position = Math.max(0, bconfig.position);
32421             fc = this.btnContainer.childNodes[bconfig.position];
32422         }
32423          
32424         var btn = new Roo.Button(
32425             fc ? 
32426                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32427                 : this.btnContainer.appendChild(document.createElement("td")),
32428             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32429             bconfig
32430         );
32431         this.syncBodyHeight();
32432         if(!this.buttons){
32433             /**
32434              * Array of all the buttons that have been added to this dialog via addButton
32435              * @type Array
32436              */
32437             this.buttons = [];
32438         }
32439         this.buttons.push(btn);
32440         return btn;
32441     },
32442
32443     /**
32444      * Sets the default button to be focused when the dialog is displayed.
32445      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32446      * @return {Roo.BasicDialog} this
32447      */
32448     setDefaultButton : function(btn){
32449         this.defaultButton = btn;
32450         return this;
32451     },
32452
32453     // private
32454     getHeaderFooterHeight : function(safe){
32455         var height = 0;
32456         if(this.header){
32457            height += this.header.getHeight();
32458         }
32459         if(this.footer){
32460            var fm = this.footer.getMargins();
32461             height += (this.footer.getHeight()+fm.top+fm.bottom);
32462         }
32463         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32464         height += this.centerBg.getPadding("tb");
32465         return height;
32466     },
32467
32468     // private
32469     syncBodyHeight : function()
32470     {
32471         var bd = this.body, // the text
32472             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32473             bw = this.bwrap;
32474         var height = this.size.height - this.getHeaderFooterHeight(false);
32475         bd.setHeight(height-bd.getMargins("tb"));
32476         var hh = this.header.getHeight();
32477         var h = this.size.height-hh;
32478         cb.setHeight(h);
32479         
32480         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32481         bw.setHeight(h-cb.getPadding("tb"));
32482         
32483         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32484         bd.setWidth(bw.getWidth(true));
32485         if(this.tabs){
32486             this.tabs.syncHeight();
32487             if(Roo.isIE){
32488                 this.tabs.el.repaint();
32489             }
32490         }
32491     },
32492
32493     /**
32494      * Restores the previous state of the dialog if Roo.state is configured.
32495      * @return {Roo.BasicDialog} this
32496      */
32497     restoreState : function(){
32498         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32499         if(box && box.width){
32500             this.xy = [box.x, box.y];
32501             this.resizeTo(box.width, box.height);
32502         }
32503         return this;
32504     },
32505
32506     // private
32507     beforeShow : function(){
32508         this.expand();
32509         if(this.fixedcenter){
32510             this.xy = this.el.getCenterXY(true);
32511         }
32512         if(this.modal){
32513             Roo.get(document.body).addClass("x-body-masked");
32514             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32515             this.mask.show();
32516         }
32517         this.constrainXY();
32518     },
32519
32520     // private
32521     animShow : function(){
32522         var b = Roo.get(this.animateTarget).getBox();
32523         this.proxy.setSize(b.width, b.height);
32524         this.proxy.setLocation(b.x, b.y);
32525         this.proxy.show();
32526         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32527                     true, .35, this.showEl.createDelegate(this));
32528     },
32529
32530     /**
32531      * Shows the dialog.
32532      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32533      * @return {Roo.BasicDialog} this
32534      */
32535     show : function(animateTarget){
32536         if (this.fireEvent("beforeshow", this) === false){
32537             return;
32538         }
32539         if(this.syncHeightBeforeShow){
32540             this.syncBodyHeight();
32541         }else if(this.firstShow){
32542             this.firstShow = false;
32543             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32544         }
32545         this.animateTarget = animateTarget || this.animateTarget;
32546         if(!this.el.isVisible()){
32547             this.beforeShow();
32548             if(this.animateTarget && Roo.get(this.animateTarget)){
32549                 this.animShow();
32550             }else{
32551                 this.showEl();
32552             }
32553         }
32554         return this;
32555     },
32556
32557     // private
32558     showEl : function(){
32559         this.proxy.hide();
32560         this.el.setXY(this.xy);
32561         this.el.show();
32562         this.adjustAssets(true);
32563         this.toFront();
32564         this.focus();
32565         // IE peekaboo bug - fix found by Dave Fenwick
32566         if(Roo.isIE){
32567             this.el.repaint();
32568         }
32569         this.fireEvent("show", this);
32570     },
32571
32572     /**
32573      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32574      * dialog itself will receive focus.
32575      */
32576     focus : function(){
32577         if(this.defaultButton){
32578             this.defaultButton.focus();
32579         }else{
32580             this.focusEl.focus();
32581         }
32582     },
32583
32584     // private
32585     constrainXY : function(){
32586         if(this.constraintoviewport !== false){
32587             if(!this.viewSize){
32588                 if(this.container){
32589                     var s = this.container.getSize();
32590                     this.viewSize = [s.width, s.height];
32591                 }else{
32592                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32593                 }
32594             }
32595             var s = Roo.get(this.container||document).getScroll();
32596
32597             var x = this.xy[0], y = this.xy[1];
32598             var w = this.size.width, h = this.size.height;
32599             var vw = this.viewSize[0], vh = this.viewSize[1];
32600             // only move it if it needs it
32601             var moved = false;
32602             // first validate right/bottom
32603             if(x + w > vw+s.left){
32604                 x = vw - w;
32605                 moved = true;
32606             }
32607             if(y + h > vh+s.top){
32608                 y = vh - h;
32609                 moved = true;
32610             }
32611             // then make sure top/left isn't negative
32612             if(x < s.left){
32613                 x = s.left;
32614                 moved = true;
32615             }
32616             if(y < s.top){
32617                 y = s.top;
32618                 moved = true;
32619             }
32620             if(moved){
32621                 // cache xy
32622                 this.xy = [x, y];
32623                 if(this.isVisible()){
32624                     this.el.setLocation(x, y);
32625                     this.adjustAssets();
32626                 }
32627             }
32628         }
32629     },
32630
32631     // private
32632     onDrag : function(){
32633         if(!this.proxyDrag){
32634             this.xy = this.el.getXY();
32635             this.adjustAssets();
32636         }
32637     },
32638
32639     // private
32640     adjustAssets : function(doShow){
32641         var x = this.xy[0], y = this.xy[1];
32642         var w = this.size.width, h = this.size.height;
32643         if(doShow === true){
32644             if(this.shadow){
32645                 this.shadow.show(this.el);
32646             }
32647             if(this.shim){
32648                 this.shim.show();
32649             }
32650         }
32651         if(this.shadow && this.shadow.isVisible()){
32652             this.shadow.show(this.el);
32653         }
32654         if(this.shim && this.shim.isVisible()){
32655             this.shim.setBounds(x, y, w, h);
32656         }
32657     },
32658
32659     // private
32660     adjustViewport : function(w, h){
32661         if(!w || !h){
32662             w = Roo.lib.Dom.getViewWidth();
32663             h = Roo.lib.Dom.getViewHeight();
32664         }
32665         // cache the size
32666         this.viewSize = [w, h];
32667         if(this.modal && this.mask.isVisible()){
32668             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32669             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32670         }
32671         if(this.isVisible()){
32672             this.constrainXY();
32673         }
32674     },
32675
32676     /**
32677      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32678      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32679      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32680      */
32681     destroy : function(removeEl){
32682         if(this.isVisible()){
32683             this.animateTarget = null;
32684             this.hide();
32685         }
32686         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32687         if(this.tabs){
32688             this.tabs.destroy(removeEl);
32689         }
32690         Roo.destroy(
32691              this.shim,
32692              this.proxy,
32693              this.resizer,
32694              this.close,
32695              this.mask
32696         );
32697         if(this.dd){
32698             this.dd.unreg();
32699         }
32700         if(this.buttons){
32701            for(var i = 0, len = this.buttons.length; i < len; i++){
32702                this.buttons[i].destroy();
32703            }
32704         }
32705         this.el.removeAllListeners();
32706         if(removeEl === true){
32707             this.el.update("");
32708             this.el.remove();
32709         }
32710         Roo.DialogManager.unregister(this);
32711     },
32712
32713     // private
32714     startMove : function(){
32715         if(this.proxyDrag){
32716             this.proxy.show();
32717         }
32718         if(this.constraintoviewport !== false){
32719             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32720         }
32721     },
32722
32723     // private
32724     endMove : function(){
32725         if(!this.proxyDrag){
32726             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32727         }else{
32728             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32729             this.proxy.hide();
32730         }
32731         this.refreshSize();
32732         this.adjustAssets();
32733         this.focus();
32734         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32735     },
32736
32737     /**
32738      * Brings this dialog to the front of any other visible dialogs
32739      * @return {Roo.BasicDialog} this
32740      */
32741     toFront : function(){
32742         Roo.DialogManager.bringToFront(this);
32743         return this;
32744     },
32745
32746     /**
32747      * Sends this dialog to the back (under) of any other visible dialogs
32748      * @return {Roo.BasicDialog} this
32749      */
32750     toBack : function(){
32751         Roo.DialogManager.sendToBack(this);
32752         return this;
32753     },
32754
32755     /**
32756      * Centers this dialog in the viewport
32757      * @return {Roo.BasicDialog} this
32758      */
32759     center : function(){
32760         var xy = this.el.getCenterXY(true);
32761         this.moveTo(xy[0], xy[1]);
32762         return this;
32763     },
32764
32765     /**
32766      * Moves the dialog's top-left corner to the specified point
32767      * @param {Number} x
32768      * @param {Number} y
32769      * @return {Roo.BasicDialog} this
32770      */
32771     moveTo : function(x, y){
32772         this.xy = [x,y];
32773         if(this.isVisible()){
32774             this.el.setXY(this.xy);
32775             this.adjustAssets();
32776         }
32777         return this;
32778     },
32779
32780     /**
32781      * Aligns the dialog to the specified element
32782      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32783      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32784      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32785      * @return {Roo.BasicDialog} this
32786      */
32787     alignTo : function(element, position, offsets){
32788         this.xy = this.el.getAlignToXY(element, position, offsets);
32789         if(this.isVisible()){
32790             this.el.setXY(this.xy);
32791             this.adjustAssets();
32792         }
32793         return this;
32794     },
32795
32796     /**
32797      * Anchors an element to another element and realigns it when the window is resized.
32798      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32799      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32800      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32801      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32802      * is a number, it is used as the buffer delay (defaults to 50ms).
32803      * @return {Roo.BasicDialog} this
32804      */
32805     anchorTo : function(el, alignment, offsets, monitorScroll){
32806         var action = function(){
32807             this.alignTo(el, alignment, offsets);
32808         };
32809         Roo.EventManager.onWindowResize(action, this);
32810         var tm = typeof monitorScroll;
32811         if(tm != 'undefined'){
32812             Roo.EventManager.on(window, 'scroll', action, this,
32813                 {buffer: tm == 'number' ? monitorScroll : 50});
32814         }
32815         action.call(this);
32816         return this;
32817     },
32818
32819     /**
32820      * Returns true if the dialog is visible
32821      * @return {Boolean}
32822      */
32823     isVisible : function(){
32824         return this.el.isVisible();
32825     },
32826
32827     // private
32828     animHide : function(callback){
32829         var b = Roo.get(this.animateTarget).getBox();
32830         this.proxy.show();
32831         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32832         this.el.hide();
32833         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32834                     this.hideEl.createDelegate(this, [callback]));
32835     },
32836
32837     /**
32838      * Hides the dialog.
32839      * @param {Function} callback (optional) Function to call when the dialog is hidden
32840      * @return {Roo.BasicDialog} this
32841      */
32842     hide : function(callback){
32843         if (this.fireEvent("beforehide", this) === false){
32844             return;
32845         }
32846         if(this.shadow){
32847             this.shadow.hide();
32848         }
32849         if(this.shim) {
32850           this.shim.hide();
32851         }
32852         // sometimes animateTarget seems to get set.. causing problems...
32853         // this just double checks..
32854         if(this.animateTarget && Roo.get(this.animateTarget)) {
32855            this.animHide(callback);
32856         }else{
32857             this.el.hide();
32858             this.hideEl(callback);
32859         }
32860         return this;
32861     },
32862
32863     // private
32864     hideEl : function(callback){
32865         this.proxy.hide();
32866         if(this.modal){
32867             this.mask.hide();
32868             Roo.get(document.body).removeClass("x-body-masked");
32869         }
32870         this.fireEvent("hide", this);
32871         if(typeof callback == "function"){
32872             callback();
32873         }
32874     },
32875
32876     // private
32877     hideAction : function(){
32878         this.setLeft("-10000px");
32879         this.setTop("-10000px");
32880         this.setStyle("visibility", "hidden");
32881     },
32882
32883     // private
32884     refreshSize : function(){
32885         this.size = this.el.getSize();
32886         this.xy = this.el.getXY();
32887         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32888     },
32889
32890     // private
32891     // z-index is managed by the DialogManager and may be overwritten at any time
32892     setZIndex : function(index){
32893         if(this.modal){
32894             this.mask.setStyle("z-index", index);
32895         }
32896         if(this.shim){
32897             this.shim.setStyle("z-index", ++index);
32898         }
32899         if(this.shadow){
32900             this.shadow.setZIndex(++index);
32901         }
32902         this.el.setStyle("z-index", ++index);
32903         if(this.proxy){
32904             this.proxy.setStyle("z-index", ++index);
32905         }
32906         if(this.resizer){
32907             this.resizer.proxy.setStyle("z-index", ++index);
32908         }
32909
32910         this.lastZIndex = index;
32911     },
32912
32913     /**
32914      * Returns the element for this dialog
32915      * @return {Roo.Element} The underlying dialog Element
32916      */
32917     getEl : function(){
32918         return this.el;
32919     }
32920 });
32921
32922 /**
32923  * @class Roo.DialogManager
32924  * Provides global access to BasicDialogs that have been created and
32925  * support for z-indexing (layering) multiple open dialogs.
32926  */
32927 Roo.DialogManager = function(){
32928     var list = {};
32929     var accessList = [];
32930     var front = null;
32931
32932     // private
32933     var sortDialogs = function(d1, d2){
32934         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32935     };
32936
32937     // private
32938     var orderDialogs = function(){
32939         accessList.sort(sortDialogs);
32940         var seed = Roo.DialogManager.zseed;
32941         for(var i = 0, len = accessList.length; i < len; i++){
32942             var dlg = accessList[i];
32943             if(dlg){
32944                 dlg.setZIndex(seed + (i*10));
32945             }
32946         }
32947     };
32948
32949     return {
32950         /**
32951          * The starting z-index for BasicDialogs (defaults to 9000)
32952          * @type Number The z-index value
32953          */
32954         zseed : 9000,
32955
32956         // private
32957         register : function(dlg){
32958             list[dlg.id] = dlg;
32959             accessList.push(dlg);
32960         },
32961
32962         // private
32963         unregister : function(dlg){
32964             delete list[dlg.id];
32965             var i=0;
32966             var len=0;
32967             if(!accessList.indexOf){
32968                 for(  i = 0, len = accessList.length; i < len; i++){
32969                     if(accessList[i] == dlg){
32970                         accessList.splice(i, 1);
32971                         return;
32972                     }
32973                 }
32974             }else{
32975                  i = accessList.indexOf(dlg);
32976                 if(i != -1){
32977                     accessList.splice(i, 1);
32978                 }
32979             }
32980         },
32981
32982         /**
32983          * Gets a registered dialog by id
32984          * @param {String/Object} id The id of the dialog or a dialog
32985          * @return {Roo.BasicDialog} this
32986          */
32987         get : function(id){
32988             return typeof id == "object" ? id : list[id];
32989         },
32990
32991         /**
32992          * Brings the specified dialog to the front
32993          * @param {String/Object} dlg The id of the dialog or a dialog
32994          * @return {Roo.BasicDialog} this
32995          */
32996         bringToFront : function(dlg){
32997             dlg = this.get(dlg);
32998             if(dlg != front){
32999                 front = dlg;
33000                 dlg._lastAccess = new Date().getTime();
33001                 orderDialogs();
33002             }
33003             return dlg;
33004         },
33005
33006         /**
33007          * Sends the specified dialog to the back
33008          * @param {String/Object} dlg The id of the dialog or a dialog
33009          * @return {Roo.BasicDialog} this
33010          */
33011         sendToBack : function(dlg){
33012             dlg = this.get(dlg);
33013             dlg._lastAccess = -(new Date().getTime());
33014             orderDialogs();
33015             return dlg;
33016         },
33017
33018         /**
33019          * Hides all dialogs
33020          */
33021         hideAll : function(){
33022             for(var id in list){
33023                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
33024                     list[id].hide();
33025                 }
33026             }
33027         }
33028     };
33029 }();
33030
33031 /**
33032  * @class Roo.LayoutDialog
33033  * @extends Roo.BasicDialog
33034  * Dialog which provides adjustments for working with a layout in a Dialog.
33035  * Add your necessary layout config options to the dialog's config.<br>
33036  * Example usage (including a nested layout):
33037  * <pre><code>
33038 if(!dialog){
33039     dialog = new Roo.LayoutDialog("download-dlg", {
33040         modal: true,
33041         width:600,
33042         height:450,
33043         shadow:true,
33044         minWidth:500,
33045         minHeight:350,
33046         autoTabs:true,
33047         proxyDrag:true,
33048         // layout config merges with the dialog config
33049         center:{
33050             tabPosition: "top",
33051             alwaysShowTabs: true
33052         }
33053     });
33054     dialog.addKeyListener(27, dialog.hide, dialog);
33055     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33056     dialog.addButton("Build It!", this.getDownload, this);
33057
33058     // we can even add nested layouts
33059     var innerLayout = new Roo.BorderLayout("dl-inner", {
33060         east: {
33061             initialSize: 200,
33062             autoScroll:true,
33063             split:true
33064         },
33065         center: {
33066             autoScroll:true
33067         }
33068     });
33069     innerLayout.beginUpdate();
33070     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33071     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33072     innerLayout.endUpdate(true);
33073
33074     var layout = dialog.getLayout();
33075     layout.beginUpdate();
33076     layout.add("center", new Roo.ContentPanel("standard-panel",
33077                         {title: "Download the Source", fitToFrame:true}));
33078     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33079                {title: "Build your own roo.js"}));
33080     layout.getRegion("center").showPanel(sp);
33081     layout.endUpdate();
33082 }
33083 </code></pre>
33084     * @constructor
33085     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33086     * @param {Object} config configuration options
33087   */
33088 Roo.LayoutDialog = function(el, cfg){
33089     
33090     var config=  cfg;
33091     if (typeof(cfg) == 'undefined') {
33092         config = Roo.apply({}, el);
33093         // not sure why we use documentElement here.. - it should always be body.
33094         // IE7 borks horribly if we use documentElement.
33095         // webkit also does not like documentElement - it creates a body element...
33096         el = Roo.get( document.body || document.documentElement ).createChild();
33097         //config.autoCreate = true;
33098     }
33099     
33100     
33101     config.autoTabs = false;
33102     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33103     this.body.setStyle({overflow:"hidden", position:"relative"});
33104     this.layout = new Roo.BorderLayout(this.body.dom, config);
33105     this.layout.monitorWindowResize = false;
33106     this.el.addClass("x-dlg-auto-layout");
33107     // fix case when center region overwrites center function
33108     this.center = Roo.BasicDialog.prototype.center;
33109     this.on("show", this.layout.layout, this.layout, true);
33110     if (config.items) {
33111         var xitems = config.items;
33112         delete config.items;
33113         Roo.each(xitems, this.addxtype, this);
33114     }
33115     
33116     
33117 };
33118 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33119     /**
33120      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33121      * @deprecated
33122      */
33123     endUpdate : function(){
33124         this.layout.endUpdate();
33125     },
33126
33127     /**
33128      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33129      *  @deprecated
33130      */
33131     beginUpdate : function(){
33132         this.layout.beginUpdate();
33133     },
33134
33135     /**
33136      * Get the BorderLayout for this dialog
33137      * @return {Roo.BorderLayout}
33138      */
33139     getLayout : function(){
33140         return this.layout;
33141     },
33142
33143     showEl : function(){
33144         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33145         if(Roo.isIE7){
33146             this.layout.layout();
33147         }
33148     },
33149
33150     // private
33151     // Use the syncHeightBeforeShow config option to control this automatically
33152     syncBodyHeight : function(){
33153         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33154         if(this.layout){this.layout.layout();}
33155     },
33156     
33157       /**
33158      * Add an xtype element (actually adds to the layout.)
33159      * @return {Object} xdata xtype object data.
33160      */
33161     
33162     addxtype : function(c) {
33163         return this.layout.addxtype(c);
33164     }
33165 });/*
33166  * Based on:
33167  * Ext JS Library 1.1.1
33168  * Copyright(c) 2006-2007, Ext JS, LLC.
33169  *
33170  * Originally Released Under LGPL - original licence link has changed is not relivant.
33171  *
33172  * Fork - LGPL
33173  * <script type="text/javascript">
33174  */
33175  
33176 /**
33177  * @class Roo.MessageBox
33178  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33179  * Example usage:
33180  *<pre><code>
33181 // Basic alert:
33182 Roo.Msg.alert('Status', 'Changes saved successfully.');
33183
33184 // Prompt for user data:
33185 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33186     if (btn == 'ok'){
33187         // process text value...
33188     }
33189 });
33190
33191 // Show a dialog using config options:
33192 Roo.Msg.show({
33193    title:'Save Changes?',
33194    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33195    buttons: Roo.Msg.YESNOCANCEL,
33196    fn: processResult,
33197    animEl: 'elId'
33198 });
33199 </code></pre>
33200  * @singleton
33201  */
33202 Roo.MessageBox = function(){
33203     var dlg, opt, mask, waitTimer;
33204     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33205     var buttons, activeTextEl, bwidth;
33206
33207     // private
33208     var handleButton = function(button){
33209         dlg.hide();
33210         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33211     };
33212
33213     // private
33214     var handleHide = function(){
33215         if(opt && opt.cls){
33216             dlg.el.removeClass(opt.cls);
33217         }
33218         if(waitTimer){
33219             Roo.TaskMgr.stop(waitTimer);
33220             waitTimer = null;
33221         }
33222     };
33223
33224     // private
33225     var updateButtons = function(b){
33226         var width = 0;
33227         if(!b){
33228             buttons["ok"].hide();
33229             buttons["cancel"].hide();
33230             buttons["yes"].hide();
33231             buttons["no"].hide();
33232             dlg.footer.dom.style.display = 'none';
33233             return width;
33234         }
33235         dlg.footer.dom.style.display = '';
33236         for(var k in buttons){
33237             if(typeof buttons[k] != "function"){
33238                 if(b[k]){
33239                     buttons[k].show();
33240                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33241                     width += buttons[k].el.getWidth()+15;
33242                 }else{
33243                     buttons[k].hide();
33244                 }
33245             }
33246         }
33247         return width;
33248     };
33249
33250     // private
33251     var handleEsc = function(d, k, e){
33252         if(opt && opt.closable !== false){
33253             dlg.hide();
33254         }
33255         if(e){
33256             e.stopEvent();
33257         }
33258     };
33259
33260     return {
33261         /**
33262          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33263          * @return {Roo.BasicDialog} The BasicDialog element
33264          */
33265         getDialog : function(){
33266            if(!dlg){
33267                 dlg = new Roo.BasicDialog("x-msg-box", {
33268                     autoCreate : true,
33269                     shadow: true,
33270                     draggable: true,
33271                     resizable:false,
33272                     constraintoviewport:false,
33273                     fixedcenter:true,
33274                     collapsible : false,
33275                     shim:true,
33276                     modal: true,
33277                     width:400, height:100,
33278                     buttonAlign:"center",
33279                     closeClick : function(){
33280                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33281                             handleButton("no");
33282                         }else{
33283                             handleButton("cancel");
33284                         }
33285                     }
33286                 });
33287                 dlg.on("hide", handleHide);
33288                 mask = dlg.mask;
33289                 dlg.addKeyListener(27, handleEsc);
33290                 buttons = {};
33291                 var bt = this.buttonText;
33292                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33293                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33294                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33295                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33296                 bodyEl = dlg.body.createChild({
33297
33298                     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>'
33299                 });
33300                 msgEl = bodyEl.dom.firstChild;
33301                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33302                 textboxEl.enableDisplayMode();
33303                 textboxEl.addKeyListener([10,13], function(){
33304                     if(dlg.isVisible() && opt && opt.buttons){
33305                         if(opt.buttons.ok){
33306                             handleButton("ok");
33307                         }else if(opt.buttons.yes){
33308                             handleButton("yes");
33309                         }
33310                     }
33311                 });
33312                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33313                 textareaEl.enableDisplayMode();
33314                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33315                 progressEl.enableDisplayMode();
33316                 var pf = progressEl.dom.firstChild;
33317                 if (pf) {
33318                     pp = Roo.get(pf.firstChild);
33319                     pp.setHeight(pf.offsetHeight);
33320                 }
33321                 
33322             }
33323             return dlg;
33324         },
33325
33326         /**
33327          * Updates the message box body text
33328          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33329          * the XHTML-compliant non-breaking space character '&amp;#160;')
33330          * @return {Roo.MessageBox} This message box
33331          */
33332         updateText : function(text){
33333             if(!dlg.isVisible() && !opt.width){
33334                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33335             }
33336             msgEl.innerHTML = text || '&#160;';
33337       
33338             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33339             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33340             var w = Math.max(
33341                     Math.min(opt.width || cw , this.maxWidth), 
33342                     Math.max(opt.minWidth || this.minWidth, bwidth)
33343             );
33344             if(opt.prompt){
33345                 activeTextEl.setWidth(w);
33346             }
33347             if(dlg.isVisible()){
33348                 dlg.fixedcenter = false;
33349             }
33350             // to big, make it scroll. = But as usual stupid IE does not support
33351             // !important..
33352             
33353             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33354                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33355                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33356             } else {
33357                 bodyEl.dom.style.height = '';
33358                 bodyEl.dom.style.overflowY = '';
33359             }
33360             if (cw > w) {
33361                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33362             } else {
33363                 bodyEl.dom.style.overflowX = '';
33364             }
33365             
33366             dlg.setContentSize(w, bodyEl.getHeight());
33367             if(dlg.isVisible()){
33368                 dlg.fixedcenter = true;
33369             }
33370             return this;
33371         },
33372
33373         /**
33374          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33375          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33376          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33377          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33378          * @return {Roo.MessageBox} This message box
33379          */
33380         updateProgress : function(value, text){
33381             if(text){
33382                 this.updateText(text);
33383             }
33384             if (pp) { // weird bug on my firefox - for some reason this is not defined
33385                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33386             }
33387             return this;
33388         },        
33389
33390         /**
33391          * Returns true if the message box is currently displayed
33392          * @return {Boolean} True if the message box is visible, else false
33393          */
33394         isVisible : function(){
33395             return dlg && dlg.isVisible();  
33396         },
33397
33398         /**
33399          * Hides the message box if it is displayed
33400          */
33401         hide : function(){
33402             if(this.isVisible()){
33403                 dlg.hide();
33404             }  
33405         },
33406
33407         /**
33408          * Displays a new message box, or reinitializes an existing message box, based on the config options
33409          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33410          * The following config object properties are supported:
33411          * <pre>
33412 Property    Type             Description
33413 ----------  ---------------  ------------------------------------------------------------------------------------
33414 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33415                                    closes (defaults to undefined)
33416 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33417                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33418 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33419                                    progress and wait dialogs will ignore this property and always hide the
33420                                    close button as they can only be closed programmatically.
33421 cls               String           A custom CSS class to apply to the message box element
33422 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33423                                    displayed (defaults to 75)
33424 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33425                                    function will be btn (the name of the button that was clicked, if applicable,
33426                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33427                                    Progress and wait dialogs will ignore this option since they do not respond to
33428                                    user actions and can only be closed programmatically, so any required function
33429                                    should be called by the same code after it closes the dialog.
33430 icon              String           A CSS class that provides a background image to be used as an icon for
33431                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33432 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33433 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33434 modal             Boolean          False to allow user interaction with the page while the message box is
33435                                    displayed (defaults to true)
33436 msg               String           A string that will replace the existing message box body text (defaults
33437                                    to the XHTML-compliant non-breaking space character '&#160;')
33438 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33439 progress          Boolean          True to display a progress bar (defaults to false)
33440 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33441 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33442 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33443 title             String           The title text
33444 value             String           The string value to set into the active textbox element if displayed
33445 wait              Boolean          True to display a progress bar (defaults to false)
33446 width             Number           The width of the dialog in pixels
33447 </pre>
33448          *
33449          * Example usage:
33450          * <pre><code>
33451 Roo.Msg.show({
33452    title: 'Address',
33453    msg: 'Please enter your address:',
33454    width: 300,
33455    buttons: Roo.MessageBox.OKCANCEL,
33456    multiline: true,
33457    fn: saveAddress,
33458    animEl: 'addAddressBtn'
33459 });
33460 </code></pre>
33461          * @param {Object} config Configuration options
33462          * @return {Roo.MessageBox} This message box
33463          */
33464         show : function(options)
33465         {
33466             
33467             // this causes nightmares if you show one dialog after another
33468             // especially on callbacks..
33469              
33470             if(this.isVisible()){
33471                 
33472                 this.hide();
33473                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33474                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33475                 Roo.log("New Dialog Message:" +  options.msg )
33476                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33477                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33478                 
33479             }
33480             var d = this.getDialog();
33481             opt = options;
33482             d.setTitle(opt.title || "&#160;");
33483             d.close.setDisplayed(opt.closable !== false);
33484             activeTextEl = textboxEl;
33485             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33486             if(opt.prompt){
33487                 if(opt.multiline){
33488                     textboxEl.hide();
33489                     textareaEl.show();
33490                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33491                         opt.multiline : this.defaultTextHeight);
33492                     activeTextEl = textareaEl;
33493                 }else{
33494                     textboxEl.show();
33495                     textareaEl.hide();
33496                 }
33497             }else{
33498                 textboxEl.hide();
33499                 textareaEl.hide();
33500             }
33501             progressEl.setDisplayed(opt.progress === true);
33502             this.updateProgress(0);
33503             activeTextEl.dom.value = opt.value || "";
33504             if(opt.prompt){
33505                 dlg.setDefaultButton(activeTextEl);
33506             }else{
33507                 var bs = opt.buttons;
33508                 var db = null;
33509                 if(bs && bs.ok){
33510                     db = buttons["ok"];
33511                 }else if(bs && bs.yes){
33512                     db = buttons["yes"];
33513                 }
33514                 dlg.setDefaultButton(db);
33515             }
33516             bwidth = updateButtons(opt.buttons);
33517             this.updateText(opt.msg);
33518             if(opt.cls){
33519                 d.el.addClass(opt.cls);
33520             }
33521             d.proxyDrag = opt.proxyDrag === true;
33522             d.modal = opt.modal !== false;
33523             d.mask = opt.modal !== false ? mask : false;
33524             if(!d.isVisible()){
33525                 // force it to the end of the z-index stack so it gets a cursor in FF
33526                 document.body.appendChild(dlg.el.dom);
33527                 d.animateTarget = null;
33528                 d.show(options.animEl);
33529             }
33530             return this;
33531         },
33532
33533         /**
33534          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33535          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33536          * and closing the message box when the process is complete.
33537          * @param {String} title The title bar text
33538          * @param {String} msg The message box body text
33539          * @return {Roo.MessageBox} This message box
33540          */
33541         progress : function(title, msg){
33542             this.show({
33543                 title : title,
33544                 msg : msg,
33545                 buttons: false,
33546                 progress:true,
33547                 closable:false,
33548                 minWidth: this.minProgressWidth,
33549                 modal : true
33550             });
33551             return this;
33552         },
33553
33554         /**
33555          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33556          * If a callback function is passed it will be called after the user clicks the button, and the
33557          * id of the button that was clicked will be passed as the only parameter to the callback
33558          * (could also be the top-right close button).
33559          * @param {String} title The title bar text
33560          * @param {String} msg The message box body text
33561          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33562          * @param {Object} scope (optional) The scope of the callback function
33563          * @return {Roo.MessageBox} This message box
33564          */
33565         alert : function(title, msg, fn, scope){
33566             this.show({
33567                 title : title,
33568                 msg : msg,
33569                 buttons: this.OK,
33570                 fn: fn,
33571                 scope : scope,
33572                 modal : true
33573             });
33574             return this;
33575         },
33576
33577         /**
33578          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33579          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33580          * You are responsible for closing the message box when the process is complete.
33581          * @param {String} msg The message box body text
33582          * @param {String} title (optional) The title bar text
33583          * @return {Roo.MessageBox} This message box
33584          */
33585         wait : function(msg, title){
33586             this.show({
33587                 title : title,
33588                 msg : msg,
33589                 buttons: false,
33590                 closable:false,
33591                 progress:true,
33592                 modal:true,
33593                 width:300,
33594                 wait:true
33595             });
33596             waitTimer = Roo.TaskMgr.start({
33597                 run: function(i){
33598                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33599                 },
33600                 interval: 1000
33601             });
33602             return this;
33603         },
33604
33605         /**
33606          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33607          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33608          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33609          * @param {String} title The title bar text
33610          * @param {String} msg The message box body text
33611          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33612          * @param {Object} scope (optional) The scope of the callback function
33613          * @return {Roo.MessageBox} This message box
33614          */
33615         confirm : function(title, msg, fn, scope){
33616             this.show({
33617                 title : title,
33618                 msg : msg,
33619                 buttons: this.YESNO,
33620                 fn: fn,
33621                 scope : scope,
33622                 modal : true
33623             });
33624             return this;
33625         },
33626
33627         /**
33628          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33629          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33630          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33631          * (could also be the top-right close button) and the text that was entered will be passed as the two
33632          * parameters to the callback.
33633          * @param {String} title The title bar text
33634          * @param {String} msg The message box body text
33635          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33636          * @param {Object} scope (optional) The scope of the callback function
33637          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33638          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33639          * @return {Roo.MessageBox} This message box
33640          */
33641         prompt : function(title, msg, fn, scope, multiline){
33642             this.show({
33643                 title : title,
33644                 msg : msg,
33645                 buttons: this.OKCANCEL,
33646                 fn: fn,
33647                 minWidth:250,
33648                 scope : scope,
33649                 prompt:true,
33650                 multiline: multiline,
33651                 modal : true
33652             });
33653             return this;
33654         },
33655
33656         /**
33657          * Button config that displays a single OK button
33658          * @type Object
33659          */
33660         OK : {ok:true},
33661         /**
33662          * Button config that displays Yes and No buttons
33663          * @type Object
33664          */
33665         YESNO : {yes:true, no:true},
33666         /**
33667          * Button config that displays OK and Cancel buttons
33668          * @type Object
33669          */
33670         OKCANCEL : {ok:true, cancel:true},
33671         /**
33672          * Button config that displays Yes, No and Cancel buttons
33673          * @type Object
33674          */
33675         YESNOCANCEL : {yes:true, no:true, cancel:true},
33676
33677         /**
33678          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33679          * @type Number
33680          */
33681         defaultTextHeight : 75,
33682         /**
33683          * The maximum width in pixels of the message box (defaults to 600)
33684          * @type Number
33685          */
33686         maxWidth : 600,
33687         /**
33688          * The minimum width in pixels of the message box (defaults to 100)
33689          * @type Number
33690          */
33691         minWidth : 100,
33692         /**
33693          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33694          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33695          * @type Number
33696          */
33697         minProgressWidth : 250,
33698         /**
33699          * An object containing the default button text strings that can be overriden for localized language support.
33700          * Supported properties are: ok, cancel, yes and no.
33701          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33702          * @type Object
33703          */
33704         buttonText : {
33705             ok : "OK",
33706             cancel : "Cancel",
33707             yes : "Yes",
33708             no : "No"
33709         }
33710     };
33711 }();
33712
33713 /**
33714  * Shorthand for {@link Roo.MessageBox}
33715  */
33716 Roo.Msg = Roo.MessageBox;/*
33717  * Based on:
33718  * Ext JS Library 1.1.1
33719  * Copyright(c) 2006-2007, Ext JS, LLC.
33720  *
33721  * Originally Released Under LGPL - original licence link has changed is not relivant.
33722  *
33723  * Fork - LGPL
33724  * <script type="text/javascript">
33725  */
33726 /**
33727  * @class Roo.QuickTips
33728  * Provides attractive and customizable tooltips for any element.
33729  * @singleton
33730  */
33731 Roo.QuickTips = function(){
33732     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33733     var ce, bd, xy, dd;
33734     var visible = false, disabled = true, inited = false;
33735     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33736     
33737     var onOver = function(e){
33738         if(disabled){
33739             return;
33740         }
33741         var t = e.getTarget();
33742         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33743             return;
33744         }
33745         if(ce && t == ce.el){
33746             clearTimeout(hideProc);
33747             return;
33748         }
33749         if(t && tagEls[t.id]){
33750             tagEls[t.id].el = t;
33751             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33752             return;
33753         }
33754         var ttp, et = Roo.fly(t);
33755         var ns = cfg.namespace;
33756         if(tm.interceptTitles && t.title){
33757             ttp = t.title;
33758             t.qtip = ttp;
33759             t.removeAttribute("title");
33760             e.preventDefault();
33761         }else{
33762             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33763         }
33764         if(ttp){
33765             showProc = show.defer(tm.showDelay, tm, [{
33766                 el: t, 
33767                 text: ttp.replace(/\\n/g,'<br/>'),
33768                 width: et.getAttributeNS(ns, cfg.width),
33769                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33770                 title: et.getAttributeNS(ns, cfg.title),
33771                     cls: et.getAttributeNS(ns, cfg.cls)
33772             }]);
33773         }
33774     };
33775     
33776     var onOut = function(e){
33777         clearTimeout(showProc);
33778         var t = e.getTarget();
33779         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33780             hideProc = setTimeout(hide, tm.hideDelay);
33781         }
33782     };
33783     
33784     var onMove = function(e){
33785         if(disabled){
33786             return;
33787         }
33788         xy = e.getXY();
33789         xy[1] += 18;
33790         if(tm.trackMouse && ce){
33791             el.setXY(xy);
33792         }
33793     };
33794     
33795     var onDown = function(e){
33796         clearTimeout(showProc);
33797         clearTimeout(hideProc);
33798         if(!e.within(el)){
33799             if(tm.hideOnClick){
33800                 hide();
33801                 tm.disable();
33802                 tm.enable.defer(100, tm);
33803             }
33804         }
33805     };
33806     
33807     var getPad = function(){
33808         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33809     };
33810
33811     var show = function(o){
33812         if(disabled){
33813             return;
33814         }
33815         clearTimeout(dismissProc);
33816         ce = o;
33817         if(removeCls){ // in case manually hidden
33818             el.removeClass(removeCls);
33819             removeCls = null;
33820         }
33821         if(ce.cls){
33822             el.addClass(ce.cls);
33823             removeCls = ce.cls;
33824         }
33825         if(ce.title){
33826             tipTitle.update(ce.title);
33827             tipTitle.show();
33828         }else{
33829             tipTitle.update('');
33830             tipTitle.hide();
33831         }
33832         el.dom.style.width  = tm.maxWidth+'px';
33833         //tipBody.dom.style.width = '';
33834         tipBodyText.update(o.text);
33835         var p = getPad(), w = ce.width;
33836         if(!w){
33837             var td = tipBodyText.dom;
33838             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33839             if(aw > tm.maxWidth){
33840                 w = tm.maxWidth;
33841             }else if(aw < tm.minWidth){
33842                 w = tm.minWidth;
33843             }else{
33844                 w = aw;
33845             }
33846         }
33847         //tipBody.setWidth(w);
33848         el.setWidth(parseInt(w, 10) + p);
33849         if(ce.autoHide === false){
33850             close.setDisplayed(true);
33851             if(dd){
33852                 dd.unlock();
33853             }
33854         }else{
33855             close.setDisplayed(false);
33856             if(dd){
33857                 dd.lock();
33858             }
33859         }
33860         if(xy){
33861             el.avoidY = xy[1]-18;
33862             el.setXY(xy);
33863         }
33864         if(tm.animate){
33865             el.setOpacity(.1);
33866             el.setStyle("visibility", "visible");
33867             el.fadeIn({callback: afterShow});
33868         }else{
33869             afterShow();
33870         }
33871     };
33872     
33873     var afterShow = function(){
33874         if(ce){
33875             el.show();
33876             esc.enable();
33877             if(tm.autoDismiss && ce.autoHide !== false){
33878                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33879             }
33880         }
33881     };
33882     
33883     var hide = function(noanim){
33884         clearTimeout(dismissProc);
33885         clearTimeout(hideProc);
33886         ce = null;
33887         if(el.isVisible()){
33888             esc.disable();
33889             if(noanim !== true && tm.animate){
33890                 el.fadeOut({callback: afterHide});
33891             }else{
33892                 afterHide();
33893             } 
33894         }
33895     };
33896     
33897     var afterHide = function(){
33898         el.hide();
33899         if(removeCls){
33900             el.removeClass(removeCls);
33901             removeCls = null;
33902         }
33903     };
33904     
33905     return {
33906         /**
33907         * @cfg {Number} minWidth
33908         * The minimum width of the quick tip (defaults to 40)
33909         */
33910        minWidth : 40,
33911         /**
33912         * @cfg {Number} maxWidth
33913         * The maximum width of the quick tip (defaults to 300)
33914         */
33915        maxWidth : 300,
33916         /**
33917         * @cfg {Boolean} interceptTitles
33918         * True to automatically use the element's DOM title value if available (defaults to false)
33919         */
33920        interceptTitles : false,
33921         /**
33922         * @cfg {Boolean} trackMouse
33923         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33924         */
33925        trackMouse : false,
33926         /**
33927         * @cfg {Boolean} hideOnClick
33928         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33929         */
33930        hideOnClick : true,
33931         /**
33932         * @cfg {Number} showDelay
33933         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33934         */
33935        showDelay : 500,
33936         /**
33937         * @cfg {Number} hideDelay
33938         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33939         */
33940        hideDelay : 200,
33941         /**
33942         * @cfg {Boolean} autoHide
33943         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33944         * Used in conjunction with hideDelay.
33945         */
33946        autoHide : true,
33947         /**
33948         * @cfg {Boolean}
33949         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33950         * (defaults to true).  Used in conjunction with autoDismissDelay.
33951         */
33952        autoDismiss : true,
33953         /**
33954         * @cfg {Number}
33955         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33956         */
33957        autoDismissDelay : 5000,
33958        /**
33959         * @cfg {Boolean} animate
33960         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33961         */
33962        animate : false,
33963
33964        /**
33965         * @cfg {String} title
33966         * Title text to display (defaults to '').  This can be any valid HTML markup.
33967         */
33968         title: '',
33969        /**
33970         * @cfg {String} text
33971         * Body text to display (defaults to '').  This can be any valid HTML markup.
33972         */
33973         text : '',
33974        /**
33975         * @cfg {String} cls
33976         * A CSS class to apply to the base quick tip element (defaults to '').
33977         */
33978         cls : '',
33979        /**
33980         * @cfg {Number} width
33981         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33982         * minWidth or maxWidth.
33983         */
33984         width : null,
33985
33986     /**
33987      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33988      * or display QuickTips in a page.
33989      */
33990        init : function(){
33991           tm = Roo.QuickTips;
33992           cfg = tm.tagConfig;
33993           if(!inited){
33994               if(!Roo.isReady){ // allow calling of init() before onReady
33995                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33996                   return;
33997               }
33998               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33999               el.fxDefaults = {stopFx: true};
34000               // maximum custom styling
34001               //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>');
34002               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>');              
34003               tipTitle = el.child('h3');
34004               tipTitle.enableDisplayMode("block");
34005               tipBody = el.child('div.x-tip-bd');
34006               tipBodyText = el.child('div.x-tip-bd-inner');
34007               //bdLeft = el.child('div.x-tip-bd-left');
34008               //bdRight = el.child('div.x-tip-bd-right');
34009               close = el.child('div.x-tip-close');
34010               close.enableDisplayMode("block");
34011               close.on("click", hide);
34012               var d = Roo.get(document);
34013               d.on("mousedown", onDown);
34014               d.on("mouseover", onOver);
34015               d.on("mouseout", onOut);
34016               d.on("mousemove", onMove);
34017               esc = d.addKeyListener(27, hide);
34018               esc.disable();
34019               if(Roo.dd.DD){
34020                   dd = el.initDD("default", null, {
34021                       onDrag : function(){
34022                           el.sync();  
34023                       }
34024                   });
34025                   dd.setHandleElId(tipTitle.id);
34026                   dd.lock();
34027               }
34028               inited = true;
34029           }
34030           this.enable(); 
34031        },
34032
34033     /**
34034      * Configures a new quick tip instance and assigns it to a target element.  The following config options
34035      * are supported:
34036      * <pre>
34037 Property    Type                   Description
34038 ----------  ---------------------  ------------------------------------------------------------------------
34039 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
34040      * </ul>
34041      * @param {Object} config The config object
34042      */
34043        register : function(config){
34044            var cs = config instanceof Array ? config : arguments;
34045            for(var i = 0, len = cs.length; i < len; i++) {
34046                var c = cs[i];
34047                var target = c.target;
34048                if(target){
34049                    if(target instanceof Array){
34050                        for(var j = 0, jlen = target.length; j < jlen; j++){
34051                            tagEls[target[j]] = c;
34052                        }
34053                    }else{
34054                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34055                    }
34056                }
34057            }
34058        },
34059
34060     /**
34061      * Removes this quick tip from its element and destroys it.
34062      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34063      */
34064        unregister : function(el){
34065            delete tagEls[Roo.id(el)];
34066        },
34067
34068     /**
34069      * Enable this quick tip.
34070      */
34071        enable : function(){
34072            if(inited && disabled){
34073                locks.pop();
34074                if(locks.length < 1){
34075                    disabled = false;
34076                }
34077            }
34078        },
34079
34080     /**
34081      * Disable this quick tip.
34082      */
34083        disable : function(){
34084           disabled = true;
34085           clearTimeout(showProc);
34086           clearTimeout(hideProc);
34087           clearTimeout(dismissProc);
34088           if(ce){
34089               hide(true);
34090           }
34091           locks.push(1);
34092        },
34093
34094     /**
34095      * Returns true if the quick tip is enabled, else false.
34096      */
34097        isEnabled : function(){
34098             return !disabled;
34099        },
34100
34101         // private
34102        tagConfig : {
34103            namespace : "roo", // was ext?? this may break..
34104            alt_namespace : "ext",
34105            attribute : "qtip",
34106            width : "width",
34107            target : "target",
34108            title : "qtitle",
34109            hide : "hide",
34110            cls : "qclass"
34111        }
34112    };
34113 }();
34114
34115 // backwards compat
34116 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34117  * Based on:
34118  * Ext JS Library 1.1.1
34119  * Copyright(c) 2006-2007, Ext JS, LLC.
34120  *
34121  * Originally Released Under LGPL - original licence link has changed is not relivant.
34122  *
34123  * Fork - LGPL
34124  * <script type="text/javascript">
34125  */
34126  
34127
34128 /**
34129  * @class Roo.tree.TreePanel
34130  * @extends Roo.data.Tree
34131
34132  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34133  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34134  * @cfg {Boolean} enableDD true to enable drag and drop
34135  * @cfg {Boolean} enableDrag true to enable just drag
34136  * @cfg {Boolean} enableDrop true to enable just drop
34137  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34138  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34139  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34140  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34141  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34142  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34143  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34144  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34145  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34146  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34147  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34148  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34149  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34150  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34151  * @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>
34152  * @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>
34153  * 
34154  * @constructor
34155  * @param {String/HTMLElement/Element} el The container element
34156  * @param {Object} config
34157  */
34158 Roo.tree.TreePanel = function(el, config){
34159     var root = false;
34160     var loader = false;
34161     if (config.root) {
34162         root = config.root;
34163         delete config.root;
34164     }
34165     if (config.loader) {
34166         loader = config.loader;
34167         delete config.loader;
34168     }
34169     
34170     Roo.apply(this, config);
34171     Roo.tree.TreePanel.superclass.constructor.call(this);
34172     this.el = Roo.get(el);
34173     this.el.addClass('x-tree');
34174     //console.log(root);
34175     if (root) {
34176         this.setRootNode( Roo.factory(root, Roo.tree));
34177     }
34178     if (loader) {
34179         this.loader = Roo.factory(loader, Roo.tree);
34180     }
34181    /**
34182     * Read-only. The id of the container element becomes this TreePanel's id.
34183     */
34184     this.id = this.el.id;
34185     this.addEvents({
34186         /**
34187         * @event beforeload
34188         * Fires before a node is loaded, return false to cancel
34189         * @param {Node} node The node being loaded
34190         */
34191         "beforeload" : true,
34192         /**
34193         * @event load
34194         * Fires when a node is loaded
34195         * @param {Node} node The node that was loaded
34196         */
34197         "load" : true,
34198         /**
34199         * @event textchange
34200         * Fires when the text for a node is changed
34201         * @param {Node} node The node
34202         * @param {String} text The new text
34203         * @param {String} oldText The old text
34204         */
34205         "textchange" : true,
34206         /**
34207         * @event beforeexpand
34208         * Fires before a node is expanded, return false to cancel.
34209         * @param {Node} node The node
34210         * @param {Boolean} deep
34211         * @param {Boolean} anim
34212         */
34213         "beforeexpand" : true,
34214         /**
34215         * @event beforecollapse
34216         * Fires before a node is collapsed, return false to cancel.
34217         * @param {Node} node The node
34218         * @param {Boolean} deep
34219         * @param {Boolean} anim
34220         */
34221         "beforecollapse" : true,
34222         /**
34223         * @event expand
34224         * Fires when a node is expanded
34225         * @param {Node} node The node
34226         */
34227         "expand" : true,
34228         /**
34229         * @event disabledchange
34230         * Fires when the disabled status of a node changes
34231         * @param {Node} node The node
34232         * @param {Boolean} disabled
34233         */
34234         "disabledchange" : true,
34235         /**
34236         * @event collapse
34237         * Fires when a node is collapsed
34238         * @param {Node} node The node
34239         */
34240         "collapse" : true,
34241         /**
34242         * @event beforeclick
34243         * Fires before click processing on a node. Return false to cancel the default action.
34244         * @param {Node} node The node
34245         * @param {Roo.EventObject} e The event object
34246         */
34247         "beforeclick":true,
34248         /**
34249         * @event checkchange
34250         * Fires when a node with a checkbox's checked property changes
34251         * @param {Node} this This node
34252         * @param {Boolean} checked
34253         */
34254         "checkchange":true,
34255         /**
34256         * @event click
34257         * Fires when a node is clicked
34258         * @param {Node} node The node
34259         * @param {Roo.EventObject} e The event object
34260         */
34261         "click":true,
34262         /**
34263         * @event dblclick
34264         * Fires when a node is double clicked
34265         * @param {Node} node The node
34266         * @param {Roo.EventObject} e The event object
34267         */
34268         "dblclick":true,
34269         /**
34270         * @event contextmenu
34271         * Fires when a node is right clicked
34272         * @param {Node} node The node
34273         * @param {Roo.EventObject} e The event object
34274         */
34275         "contextmenu":true,
34276         /**
34277         * @event beforechildrenrendered
34278         * Fires right before the child nodes for a node are rendered
34279         * @param {Node} node The node
34280         */
34281         "beforechildrenrendered":true,
34282         /**
34283         * @event startdrag
34284         * Fires when a node starts being dragged
34285         * @param {Roo.tree.TreePanel} this
34286         * @param {Roo.tree.TreeNode} node
34287         * @param {event} e The raw browser event
34288         */ 
34289        "startdrag" : true,
34290        /**
34291         * @event enddrag
34292         * Fires when a drag operation is complete
34293         * @param {Roo.tree.TreePanel} this
34294         * @param {Roo.tree.TreeNode} node
34295         * @param {event} e The raw browser event
34296         */
34297        "enddrag" : true,
34298        /**
34299         * @event dragdrop
34300         * Fires when a dragged node is dropped on a valid DD target
34301         * @param {Roo.tree.TreePanel} this
34302         * @param {Roo.tree.TreeNode} node
34303         * @param {DD} dd The dd it was dropped on
34304         * @param {event} e The raw browser event
34305         */
34306        "dragdrop" : true,
34307        /**
34308         * @event beforenodedrop
34309         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34310         * passed to handlers has the following properties:<br />
34311         * <ul style="padding:5px;padding-left:16px;">
34312         * <li>tree - The TreePanel</li>
34313         * <li>target - The node being targeted for the drop</li>
34314         * <li>data - The drag data from the drag source</li>
34315         * <li>point - The point of the drop - append, above or below</li>
34316         * <li>source - The drag source</li>
34317         * <li>rawEvent - Raw mouse event</li>
34318         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34319         * to be inserted by setting them on this object.</li>
34320         * <li>cancel - Set this to true to cancel the drop.</li>
34321         * </ul>
34322         * @param {Object} dropEvent
34323         */
34324        "beforenodedrop" : true,
34325        /**
34326         * @event nodedrop
34327         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34328         * passed to handlers has the following properties:<br />
34329         * <ul style="padding:5px;padding-left:16px;">
34330         * <li>tree - The TreePanel</li>
34331         * <li>target - The node being targeted for the drop</li>
34332         * <li>data - The drag data from the drag source</li>
34333         * <li>point - The point of the drop - append, above or below</li>
34334         * <li>source - The drag source</li>
34335         * <li>rawEvent - Raw mouse event</li>
34336         * <li>dropNode - Dropped node(s).</li>
34337         * </ul>
34338         * @param {Object} dropEvent
34339         */
34340        "nodedrop" : true,
34341         /**
34342         * @event nodedragover
34343         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34344         * passed to handlers has the following properties:<br />
34345         * <ul style="padding:5px;padding-left:16px;">
34346         * <li>tree - The TreePanel</li>
34347         * <li>target - The node being targeted for the drop</li>
34348         * <li>data - The drag data from the drag source</li>
34349         * <li>point - The point of the drop - append, above or below</li>
34350         * <li>source - The drag source</li>
34351         * <li>rawEvent - Raw mouse event</li>
34352         * <li>dropNode - Drop node(s) provided by the source.</li>
34353         * <li>cancel - Set this to true to signal drop not allowed.</li>
34354         * </ul>
34355         * @param {Object} dragOverEvent
34356         */
34357        "nodedragover" : true,
34358        /**
34359         * @event appendnode
34360         * Fires when append node to the tree
34361         * @param {Roo.tree.TreePanel} this
34362         * @param {Roo.tree.TreeNode} node
34363         * @param {Number} index The index of the newly appended node
34364         */
34365        "appendnode" : true
34366         
34367     });
34368     if(this.singleExpand){
34369        this.on("beforeexpand", this.restrictExpand, this);
34370     }
34371     if (this.editor) {
34372         this.editor.tree = this;
34373         this.editor = Roo.factory(this.editor, Roo.tree);
34374     }
34375     
34376     if (this.selModel) {
34377         this.selModel = Roo.factory(this.selModel, Roo.tree);
34378     }
34379    
34380 };
34381 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34382     rootVisible : true,
34383     animate: Roo.enableFx,
34384     lines : true,
34385     enableDD : false,
34386     hlDrop : Roo.enableFx,
34387   
34388     renderer: false,
34389     
34390     rendererTip: false,
34391     // private
34392     restrictExpand : function(node){
34393         var p = node.parentNode;
34394         if(p){
34395             if(p.expandedChild && p.expandedChild.parentNode == p){
34396                 p.expandedChild.collapse();
34397             }
34398             p.expandedChild = node;
34399         }
34400     },
34401
34402     // private override
34403     setRootNode : function(node){
34404         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34405         if(!this.rootVisible){
34406             node.ui = new Roo.tree.RootTreeNodeUI(node);
34407         }
34408         return node;
34409     },
34410
34411     /**
34412      * Returns the container element for this TreePanel
34413      */
34414     getEl : function(){
34415         return this.el;
34416     },
34417
34418     /**
34419      * Returns the default TreeLoader for this TreePanel
34420      */
34421     getLoader : function(){
34422         return this.loader;
34423     },
34424
34425     /**
34426      * Expand all nodes
34427      */
34428     expandAll : function(){
34429         this.root.expand(true);
34430     },
34431
34432     /**
34433      * Collapse all nodes
34434      */
34435     collapseAll : function(){
34436         this.root.collapse(true);
34437     },
34438
34439     /**
34440      * Returns the selection model used by this TreePanel
34441      */
34442     getSelectionModel : function(){
34443         if(!this.selModel){
34444             this.selModel = new Roo.tree.DefaultSelectionModel();
34445         }
34446         return this.selModel;
34447     },
34448
34449     /**
34450      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34451      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34452      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34453      * @return {Array}
34454      */
34455     getChecked : function(a, startNode){
34456         startNode = startNode || this.root;
34457         var r = [];
34458         var f = function(){
34459             if(this.attributes.checked){
34460                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34461             }
34462         }
34463         startNode.cascade(f);
34464         return r;
34465     },
34466
34467     /**
34468      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34469      * @param {String} path
34470      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34471      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34472      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34473      */
34474     expandPath : function(path, attr, callback){
34475         attr = attr || "id";
34476         var keys = path.split(this.pathSeparator);
34477         var curNode = this.root;
34478         if(curNode.attributes[attr] != keys[1]){ // invalid root
34479             if(callback){
34480                 callback(false, null);
34481             }
34482             return;
34483         }
34484         var index = 1;
34485         var f = function(){
34486             if(++index == keys.length){
34487                 if(callback){
34488                     callback(true, curNode);
34489                 }
34490                 return;
34491             }
34492             var c = curNode.findChild(attr, keys[index]);
34493             if(!c){
34494                 if(callback){
34495                     callback(false, curNode);
34496                 }
34497                 return;
34498             }
34499             curNode = c;
34500             c.expand(false, false, f);
34501         };
34502         curNode.expand(false, false, f);
34503     },
34504
34505     /**
34506      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34507      * @param {String} path
34508      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34509      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34510      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34511      */
34512     selectPath : function(path, attr, callback){
34513         attr = attr || "id";
34514         var keys = path.split(this.pathSeparator);
34515         var v = keys.pop();
34516         if(keys.length > 0){
34517             var f = function(success, node){
34518                 if(success && node){
34519                     var n = node.findChild(attr, v);
34520                     if(n){
34521                         n.select();
34522                         if(callback){
34523                             callback(true, n);
34524                         }
34525                     }else if(callback){
34526                         callback(false, n);
34527                     }
34528                 }else{
34529                     if(callback){
34530                         callback(false, n);
34531                     }
34532                 }
34533             };
34534             this.expandPath(keys.join(this.pathSeparator), attr, f);
34535         }else{
34536             this.root.select();
34537             if(callback){
34538                 callback(true, this.root);
34539             }
34540         }
34541     },
34542
34543     getTreeEl : function(){
34544         return this.el;
34545     },
34546
34547     /**
34548      * Trigger rendering of this TreePanel
34549      */
34550     render : function(){
34551         if (this.innerCt) {
34552             return this; // stop it rendering more than once!!
34553         }
34554         
34555         this.innerCt = this.el.createChild({tag:"ul",
34556                cls:"x-tree-root-ct " +
34557                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34558
34559         if(this.containerScroll){
34560             Roo.dd.ScrollManager.register(this.el);
34561         }
34562         if((this.enableDD || this.enableDrop) && !this.dropZone){
34563            /**
34564             * The dropZone used by this tree if drop is enabled
34565             * @type Roo.tree.TreeDropZone
34566             */
34567              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34568                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34569            });
34570         }
34571         if((this.enableDD || this.enableDrag) && !this.dragZone){
34572            /**
34573             * The dragZone used by this tree if drag is enabled
34574             * @type Roo.tree.TreeDragZone
34575             */
34576             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34577                ddGroup: this.ddGroup || "TreeDD",
34578                scroll: this.ddScroll
34579            });
34580         }
34581         this.getSelectionModel().init(this);
34582         if (!this.root) {
34583             Roo.log("ROOT not set in tree");
34584             return this;
34585         }
34586         this.root.render();
34587         if(!this.rootVisible){
34588             this.root.renderChildren();
34589         }
34590         return this;
34591     }
34592 });/*
34593  * Based on:
34594  * Ext JS Library 1.1.1
34595  * Copyright(c) 2006-2007, Ext JS, LLC.
34596  *
34597  * Originally Released Under LGPL - original licence link has changed is not relivant.
34598  *
34599  * Fork - LGPL
34600  * <script type="text/javascript">
34601  */
34602  
34603
34604 /**
34605  * @class Roo.tree.DefaultSelectionModel
34606  * @extends Roo.util.Observable
34607  * The default single selection for a TreePanel.
34608  * @param {Object} cfg Configuration
34609  */
34610 Roo.tree.DefaultSelectionModel = function(cfg){
34611    this.selNode = null;
34612    
34613    
34614    
34615    this.addEvents({
34616        /**
34617         * @event selectionchange
34618         * Fires when the selected node changes
34619         * @param {DefaultSelectionModel} this
34620         * @param {TreeNode} node the new selection
34621         */
34622        "selectionchange" : true,
34623
34624        /**
34625         * @event beforeselect
34626         * Fires before the selected node changes, return false to cancel the change
34627         * @param {DefaultSelectionModel} this
34628         * @param {TreeNode} node the new selection
34629         * @param {TreeNode} node the old selection
34630         */
34631        "beforeselect" : true
34632    });
34633    
34634     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34635 };
34636
34637 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34638     init : function(tree){
34639         this.tree = tree;
34640         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34641         tree.on("click", this.onNodeClick, this);
34642     },
34643     
34644     onNodeClick : function(node, e){
34645         if (e.ctrlKey && this.selNode == node)  {
34646             this.unselect(node);
34647             return;
34648         }
34649         this.select(node);
34650     },
34651     
34652     /**
34653      * Select a node.
34654      * @param {TreeNode} node The node to select
34655      * @return {TreeNode} The selected node
34656      */
34657     select : function(node){
34658         var last = this.selNode;
34659         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34660             if(last){
34661                 last.ui.onSelectedChange(false);
34662             }
34663             this.selNode = node;
34664             node.ui.onSelectedChange(true);
34665             this.fireEvent("selectionchange", this, node, last);
34666         }
34667         return node;
34668     },
34669     
34670     /**
34671      * Deselect a node.
34672      * @param {TreeNode} node The node to unselect
34673      */
34674     unselect : function(node){
34675         if(this.selNode == node){
34676             this.clearSelections();
34677         }    
34678     },
34679     
34680     /**
34681      * Clear all selections
34682      */
34683     clearSelections : function(){
34684         var n = this.selNode;
34685         if(n){
34686             n.ui.onSelectedChange(false);
34687             this.selNode = null;
34688             this.fireEvent("selectionchange", this, null);
34689         }
34690         return n;
34691     },
34692     
34693     /**
34694      * Get the selected node
34695      * @return {TreeNode} The selected node
34696      */
34697     getSelectedNode : function(){
34698         return this.selNode;    
34699     },
34700     
34701     /**
34702      * Returns true if the node is selected
34703      * @param {TreeNode} node The node to check
34704      * @return {Boolean}
34705      */
34706     isSelected : function(node){
34707         return this.selNode == node;  
34708     },
34709
34710     /**
34711      * Selects the node above the selected node in the tree, intelligently walking the nodes
34712      * @return TreeNode The new selection
34713      */
34714     selectPrevious : function(){
34715         var s = this.selNode || this.lastSelNode;
34716         if(!s){
34717             return null;
34718         }
34719         var ps = s.previousSibling;
34720         if(ps){
34721             if(!ps.isExpanded() || ps.childNodes.length < 1){
34722                 return this.select(ps);
34723             } else{
34724                 var lc = ps.lastChild;
34725                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34726                     lc = lc.lastChild;
34727                 }
34728                 return this.select(lc);
34729             }
34730         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34731             return this.select(s.parentNode);
34732         }
34733         return null;
34734     },
34735
34736     /**
34737      * Selects the node above the selected node in the tree, intelligently walking the nodes
34738      * @return TreeNode The new selection
34739      */
34740     selectNext : function(){
34741         var s = this.selNode || this.lastSelNode;
34742         if(!s){
34743             return null;
34744         }
34745         if(s.firstChild && s.isExpanded()){
34746              return this.select(s.firstChild);
34747          }else if(s.nextSibling){
34748              return this.select(s.nextSibling);
34749          }else if(s.parentNode){
34750             var newS = null;
34751             s.parentNode.bubble(function(){
34752                 if(this.nextSibling){
34753                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34754                     return false;
34755                 }
34756             });
34757             return newS;
34758          }
34759         return null;
34760     },
34761
34762     onKeyDown : function(e){
34763         var s = this.selNode || this.lastSelNode;
34764         // undesirable, but required
34765         var sm = this;
34766         if(!s){
34767             return;
34768         }
34769         var k = e.getKey();
34770         switch(k){
34771              case e.DOWN:
34772                  e.stopEvent();
34773                  this.selectNext();
34774              break;
34775              case e.UP:
34776                  e.stopEvent();
34777                  this.selectPrevious();
34778              break;
34779              case e.RIGHT:
34780                  e.preventDefault();
34781                  if(s.hasChildNodes()){
34782                      if(!s.isExpanded()){
34783                          s.expand();
34784                      }else if(s.firstChild){
34785                          this.select(s.firstChild, e);
34786                      }
34787                  }
34788              break;
34789              case e.LEFT:
34790                  e.preventDefault();
34791                  if(s.hasChildNodes() && s.isExpanded()){
34792                      s.collapse();
34793                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34794                      this.select(s.parentNode, e);
34795                  }
34796              break;
34797         };
34798     }
34799 });
34800
34801 /**
34802  * @class Roo.tree.MultiSelectionModel
34803  * @extends Roo.util.Observable
34804  * Multi selection for a TreePanel.
34805  * @param {Object} cfg Configuration
34806  */
34807 Roo.tree.MultiSelectionModel = function(){
34808    this.selNodes = [];
34809    this.selMap = {};
34810    this.addEvents({
34811        /**
34812         * @event selectionchange
34813         * Fires when the selected nodes change
34814         * @param {MultiSelectionModel} this
34815         * @param {Array} nodes Array of the selected nodes
34816         */
34817        "selectionchange" : true
34818    });
34819    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34820    
34821 };
34822
34823 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34824     init : function(tree){
34825         this.tree = tree;
34826         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34827         tree.on("click", this.onNodeClick, this);
34828     },
34829     
34830     onNodeClick : function(node, e){
34831         this.select(node, e, e.ctrlKey);
34832     },
34833     
34834     /**
34835      * Select a node.
34836      * @param {TreeNode} node The node to select
34837      * @param {EventObject} e (optional) An event associated with the selection
34838      * @param {Boolean} keepExisting True to retain existing selections
34839      * @return {TreeNode} The selected node
34840      */
34841     select : function(node, e, keepExisting){
34842         if(keepExisting !== true){
34843             this.clearSelections(true);
34844         }
34845         if(this.isSelected(node)){
34846             this.lastSelNode = node;
34847             return node;
34848         }
34849         this.selNodes.push(node);
34850         this.selMap[node.id] = node;
34851         this.lastSelNode = node;
34852         node.ui.onSelectedChange(true);
34853         this.fireEvent("selectionchange", this, this.selNodes);
34854         return node;
34855     },
34856     
34857     /**
34858      * Deselect a node.
34859      * @param {TreeNode} node The node to unselect
34860      */
34861     unselect : function(node){
34862         if(this.selMap[node.id]){
34863             node.ui.onSelectedChange(false);
34864             var sn = this.selNodes;
34865             var index = -1;
34866             if(sn.indexOf){
34867                 index = sn.indexOf(node);
34868             }else{
34869                 for(var i = 0, len = sn.length; i < len; i++){
34870                     if(sn[i] == node){
34871                         index = i;
34872                         break;
34873                     }
34874                 }
34875             }
34876             if(index != -1){
34877                 this.selNodes.splice(index, 1);
34878             }
34879             delete this.selMap[node.id];
34880             this.fireEvent("selectionchange", this, this.selNodes);
34881         }
34882     },
34883     
34884     /**
34885      * Clear all selections
34886      */
34887     clearSelections : function(suppressEvent){
34888         var sn = this.selNodes;
34889         if(sn.length > 0){
34890             for(var i = 0, len = sn.length; i < len; i++){
34891                 sn[i].ui.onSelectedChange(false);
34892             }
34893             this.selNodes = [];
34894             this.selMap = {};
34895             if(suppressEvent !== true){
34896                 this.fireEvent("selectionchange", this, this.selNodes);
34897             }
34898         }
34899     },
34900     
34901     /**
34902      * Returns true if the node is selected
34903      * @param {TreeNode} node The node to check
34904      * @return {Boolean}
34905      */
34906     isSelected : function(node){
34907         return this.selMap[node.id] ? true : false;  
34908     },
34909     
34910     /**
34911      * Returns an array of the selected nodes
34912      * @return {Array}
34913      */
34914     getSelectedNodes : function(){
34915         return this.selNodes;    
34916     },
34917
34918     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34919
34920     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34921
34922     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34923 });/*
34924  * Based on:
34925  * Ext JS Library 1.1.1
34926  * Copyright(c) 2006-2007, Ext JS, LLC.
34927  *
34928  * Originally Released Under LGPL - original licence link has changed is not relivant.
34929  *
34930  * Fork - LGPL
34931  * <script type="text/javascript">
34932  */
34933  
34934 /**
34935  * @class Roo.tree.TreeNode
34936  * @extends Roo.data.Node
34937  * @cfg {String} text The text for this node
34938  * @cfg {Boolean} expanded true to start the node expanded
34939  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34940  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34941  * @cfg {Boolean} disabled true to start the node disabled
34942  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34943  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34944  * @cfg {String} cls A css class to be added to the node
34945  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34946  * @cfg {String} href URL of the link used for the node (defaults to #)
34947  * @cfg {String} hrefTarget target frame for the link
34948  * @cfg {String} qtip An Ext QuickTip for the node
34949  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34950  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34951  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34952  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34953  * (defaults to undefined with no checkbox rendered)
34954  * @constructor
34955  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34956  */
34957 Roo.tree.TreeNode = function(attributes){
34958     attributes = attributes || {};
34959     if(typeof attributes == "string"){
34960         attributes = {text: attributes};
34961     }
34962     this.childrenRendered = false;
34963     this.rendered = false;
34964     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34965     this.expanded = attributes.expanded === true;
34966     this.isTarget = attributes.isTarget !== false;
34967     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34968     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34969
34970     /**
34971      * Read-only. The text for this node. To change it use setText().
34972      * @type String
34973      */
34974     this.text = attributes.text;
34975     /**
34976      * True if this node is disabled.
34977      * @type Boolean
34978      */
34979     this.disabled = attributes.disabled === true;
34980
34981     this.addEvents({
34982         /**
34983         * @event textchange
34984         * Fires when the text for this node is changed
34985         * @param {Node} this This node
34986         * @param {String} text The new text
34987         * @param {String} oldText The old text
34988         */
34989         "textchange" : true,
34990         /**
34991         * @event beforeexpand
34992         * Fires before this node is expanded, return false to cancel.
34993         * @param {Node} this This node
34994         * @param {Boolean} deep
34995         * @param {Boolean} anim
34996         */
34997         "beforeexpand" : true,
34998         /**
34999         * @event beforecollapse
35000         * Fires before this node is collapsed, return false to cancel.
35001         * @param {Node} this This node
35002         * @param {Boolean} deep
35003         * @param {Boolean} anim
35004         */
35005         "beforecollapse" : true,
35006         /**
35007         * @event expand
35008         * Fires when this node is expanded
35009         * @param {Node} this This node
35010         */
35011         "expand" : true,
35012         /**
35013         * @event disabledchange
35014         * Fires when the disabled status of this node changes
35015         * @param {Node} this This node
35016         * @param {Boolean} disabled
35017         */
35018         "disabledchange" : true,
35019         /**
35020         * @event collapse
35021         * Fires when this node is collapsed
35022         * @param {Node} this This node
35023         */
35024         "collapse" : true,
35025         /**
35026         * @event beforeclick
35027         * Fires before click processing. Return false to cancel the default action.
35028         * @param {Node} this This node
35029         * @param {Roo.EventObject} e The event object
35030         */
35031         "beforeclick":true,
35032         /**
35033         * @event checkchange
35034         * Fires when a node with a checkbox's checked property changes
35035         * @param {Node} this This node
35036         * @param {Boolean} checked
35037         */
35038         "checkchange":true,
35039         /**
35040         * @event click
35041         * Fires when this node is clicked
35042         * @param {Node} this This node
35043         * @param {Roo.EventObject} e The event object
35044         */
35045         "click":true,
35046         /**
35047         * @event dblclick
35048         * Fires when this node is double clicked
35049         * @param {Node} this This node
35050         * @param {Roo.EventObject} e The event object
35051         */
35052         "dblclick":true,
35053         /**
35054         * @event contextmenu
35055         * Fires when this node is right clicked
35056         * @param {Node} this This node
35057         * @param {Roo.EventObject} e The event object
35058         */
35059         "contextmenu":true,
35060         /**
35061         * @event beforechildrenrendered
35062         * Fires right before the child nodes for this node are rendered
35063         * @param {Node} this This node
35064         */
35065         "beforechildrenrendered":true
35066     });
35067
35068     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35069
35070     /**
35071      * Read-only. The UI for this node
35072      * @type TreeNodeUI
35073      */
35074     this.ui = new uiClass(this);
35075     
35076     // finally support items[]
35077     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35078         return;
35079     }
35080     
35081     
35082     Roo.each(this.attributes.items, function(c) {
35083         this.appendChild(Roo.factory(c,Roo.Tree));
35084     }, this);
35085     delete this.attributes.items;
35086     
35087     
35088     
35089 };
35090 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35091     preventHScroll: true,
35092     /**
35093      * Returns true if this node is expanded
35094      * @return {Boolean}
35095      */
35096     isExpanded : function(){
35097         return this.expanded;
35098     },
35099
35100     /**
35101      * Returns the UI object for this node
35102      * @return {TreeNodeUI}
35103      */
35104     getUI : function(){
35105         return this.ui;
35106     },
35107
35108     // private override
35109     setFirstChild : function(node){
35110         var of = this.firstChild;
35111         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35112         if(this.childrenRendered && of && node != of){
35113             of.renderIndent(true, true);
35114         }
35115         if(this.rendered){
35116             this.renderIndent(true, true);
35117         }
35118     },
35119
35120     // private override
35121     setLastChild : function(node){
35122         var ol = this.lastChild;
35123         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35124         if(this.childrenRendered && ol && node != ol){
35125             ol.renderIndent(true, true);
35126         }
35127         if(this.rendered){
35128             this.renderIndent(true, true);
35129         }
35130     },
35131
35132     // these methods are overridden to provide lazy rendering support
35133     // private override
35134     appendChild : function()
35135     {
35136         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35137         if(node && this.childrenRendered){
35138             node.render();
35139         }
35140         this.ui.updateExpandIcon();
35141         return node;
35142     },
35143
35144     // private override
35145     removeChild : function(node){
35146         this.ownerTree.getSelectionModel().unselect(node);
35147         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35148         // if it's been rendered remove dom node
35149         if(this.childrenRendered){
35150             node.ui.remove();
35151         }
35152         if(this.childNodes.length < 1){
35153             this.collapse(false, false);
35154         }else{
35155             this.ui.updateExpandIcon();
35156         }
35157         if(!this.firstChild) {
35158             this.childrenRendered = false;
35159         }
35160         return node;
35161     },
35162
35163     // private override
35164     insertBefore : function(node, refNode){
35165         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35166         if(newNode && refNode && this.childrenRendered){
35167             node.render();
35168         }
35169         this.ui.updateExpandIcon();
35170         return newNode;
35171     },
35172
35173     /**
35174      * Sets the text for this node
35175      * @param {String} text
35176      */
35177     setText : function(text){
35178         var oldText = this.text;
35179         this.text = text;
35180         this.attributes.text = text;
35181         if(this.rendered){ // event without subscribing
35182             this.ui.onTextChange(this, text, oldText);
35183         }
35184         this.fireEvent("textchange", this, text, oldText);
35185     },
35186
35187     /**
35188      * Triggers selection of this node
35189      */
35190     select : function(){
35191         this.getOwnerTree().getSelectionModel().select(this);
35192     },
35193
35194     /**
35195      * Triggers deselection of this node
35196      */
35197     unselect : function(){
35198         this.getOwnerTree().getSelectionModel().unselect(this);
35199     },
35200
35201     /**
35202      * Returns true if this node is selected
35203      * @return {Boolean}
35204      */
35205     isSelected : function(){
35206         return this.getOwnerTree().getSelectionModel().isSelected(this);
35207     },
35208
35209     /**
35210      * Expand this node.
35211      * @param {Boolean} deep (optional) True to expand all children as well
35212      * @param {Boolean} anim (optional) false to cancel the default animation
35213      * @param {Function} callback (optional) A callback to be called when
35214      * expanding this node completes (does not wait for deep expand to complete).
35215      * Called with 1 parameter, this node.
35216      */
35217     expand : function(deep, anim, callback){
35218         if(!this.expanded){
35219             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35220                 return;
35221             }
35222             if(!this.childrenRendered){
35223                 this.renderChildren();
35224             }
35225             this.expanded = true;
35226             
35227             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35228                 this.ui.animExpand(function(){
35229                     this.fireEvent("expand", this);
35230                     if(typeof callback == "function"){
35231                         callback(this);
35232                     }
35233                     if(deep === true){
35234                         this.expandChildNodes(true);
35235                     }
35236                 }.createDelegate(this));
35237                 return;
35238             }else{
35239                 this.ui.expand();
35240                 this.fireEvent("expand", this);
35241                 if(typeof callback == "function"){
35242                     callback(this);
35243                 }
35244             }
35245         }else{
35246            if(typeof callback == "function"){
35247                callback(this);
35248            }
35249         }
35250         if(deep === true){
35251             this.expandChildNodes(true);
35252         }
35253     },
35254
35255     isHiddenRoot : function(){
35256         return this.isRoot && !this.getOwnerTree().rootVisible;
35257     },
35258
35259     /**
35260      * Collapse this node.
35261      * @param {Boolean} deep (optional) True to collapse all children as well
35262      * @param {Boolean} anim (optional) false to cancel the default animation
35263      */
35264     collapse : function(deep, anim){
35265         if(this.expanded && !this.isHiddenRoot()){
35266             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35267                 return;
35268             }
35269             this.expanded = false;
35270             if((this.getOwnerTree().animate && anim !== false) || anim){
35271                 this.ui.animCollapse(function(){
35272                     this.fireEvent("collapse", this);
35273                     if(deep === true){
35274                         this.collapseChildNodes(true);
35275                     }
35276                 }.createDelegate(this));
35277                 return;
35278             }else{
35279                 this.ui.collapse();
35280                 this.fireEvent("collapse", this);
35281             }
35282         }
35283         if(deep === true){
35284             var cs = this.childNodes;
35285             for(var i = 0, len = cs.length; i < len; i++) {
35286                 cs[i].collapse(true, false);
35287             }
35288         }
35289     },
35290
35291     // private
35292     delayedExpand : function(delay){
35293         if(!this.expandProcId){
35294             this.expandProcId = this.expand.defer(delay, this);
35295         }
35296     },
35297
35298     // private
35299     cancelExpand : function(){
35300         if(this.expandProcId){
35301             clearTimeout(this.expandProcId);
35302         }
35303         this.expandProcId = false;
35304     },
35305
35306     /**
35307      * Toggles expanded/collapsed state of the node
35308      */
35309     toggle : function(){
35310         if(this.expanded){
35311             this.collapse();
35312         }else{
35313             this.expand();
35314         }
35315     },
35316
35317     /**
35318      * Ensures all parent nodes are expanded
35319      */
35320     ensureVisible : function(callback){
35321         var tree = this.getOwnerTree();
35322         tree.expandPath(this.parentNode.getPath(), false, function(){
35323             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35324             Roo.callback(callback);
35325         }.createDelegate(this));
35326     },
35327
35328     /**
35329      * Expand all child nodes
35330      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35331      */
35332     expandChildNodes : function(deep){
35333         var cs = this.childNodes;
35334         for(var i = 0, len = cs.length; i < len; i++) {
35335                 cs[i].expand(deep);
35336         }
35337     },
35338
35339     /**
35340      * Collapse all child nodes
35341      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35342      */
35343     collapseChildNodes : function(deep){
35344         var cs = this.childNodes;
35345         for(var i = 0, len = cs.length; i < len; i++) {
35346                 cs[i].collapse(deep);
35347         }
35348     },
35349
35350     /**
35351      * Disables this node
35352      */
35353     disable : function(){
35354         this.disabled = true;
35355         this.unselect();
35356         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35357             this.ui.onDisableChange(this, true);
35358         }
35359         this.fireEvent("disabledchange", this, true);
35360     },
35361
35362     /**
35363      * Enables this node
35364      */
35365     enable : function(){
35366         this.disabled = false;
35367         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35368             this.ui.onDisableChange(this, false);
35369         }
35370         this.fireEvent("disabledchange", this, false);
35371     },
35372
35373     // private
35374     renderChildren : function(suppressEvent){
35375         if(suppressEvent !== false){
35376             this.fireEvent("beforechildrenrendered", this);
35377         }
35378         var cs = this.childNodes;
35379         for(var i = 0, len = cs.length; i < len; i++){
35380             cs[i].render(true);
35381         }
35382         this.childrenRendered = true;
35383     },
35384
35385     // private
35386     sort : function(fn, scope){
35387         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35388         if(this.childrenRendered){
35389             var cs = this.childNodes;
35390             for(var i = 0, len = cs.length; i < len; i++){
35391                 cs[i].render(true);
35392             }
35393         }
35394     },
35395
35396     // private
35397     render : function(bulkRender){
35398         this.ui.render(bulkRender);
35399         if(!this.rendered){
35400             this.rendered = true;
35401             if(this.expanded){
35402                 this.expanded = false;
35403                 this.expand(false, false);
35404             }
35405         }
35406     },
35407
35408     // private
35409     renderIndent : function(deep, refresh){
35410         if(refresh){
35411             this.ui.childIndent = null;
35412         }
35413         this.ui.renderIndent();
35414         if(deep === true && this.childrenRendered){
35415             var cs = this.childNodes;
35416             for(var i = 0, len = cs.length; i < len; i++){
35417                 cs[i].renderIndent(true, refresh);
35418             }
35419         }
35420     }
35421 });/*
35422  * Based on:
35423  * Ext JS Library 1.1.1
35424  * Copyright(c) 2006-2007, Ext JS, LLC.
35425  *
35426  * Originally Released Under LGPL - original licence link has changed is not relivant.
35427  *
35428  * Fork - LGPL
35429  * <script type="text/javascript">
35430  */
35431  
35432 /**
35433  * @class Roo.tree.AsyncTreeNode
35434  * @extends Roo.tree.TreeNode
35435  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35436  * @constructor
35437  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35438  */
35439  Roo.tree.AsyncTreeNode = function(config){
35440     this.loaded = false;
35441     this.loading = false;
35442     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35443     /**
35444     * @event beforeload
35445     * Fires before this node is loaded, return false to cancel
35446     * @param {Node} this This node
35447     */
35448     this.addEvents({'beforeload':true, 'load': true});
35449     /**
35450     * @event load
35451     * Fires when this node is loaded
35452     * @param {Node} this This node
35453     */
35454     /**
35455      * The loader used by this node (defaults to using the tree's defined loader)
35456      * @type TreeLoader
35457      * @property loader
35458      */
35459 };
35460 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35461     expand : function(deep, anim, callback){
35462         if(this.loading){ // if an async load is already running, waiting til it's done
35463             var timer;
35464             var f = function(){
35465                 if(!this.loading){ // done loading
35466                     clearInterval(timer);
35467                     this.expand(deep, anim, callback);
35468                 }
35469             }.createDelegate(this);
35470             timer = setInterval(f, 200);
35471             return;
35472         }
35473         if(!this.loaded){
35474             if(this.fireEvent("beforeload", this) === false){
35475                 return;
35476             }
35477             this.loading = true;
35478             this.ui.beforeLoad(this);
35479             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35480             if(loader){
35481                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35482                 return;
35483             }
35484         }
35485         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35486     },
35487     
35488     /**
35489      * Returns true if this node is currently loading
35490      * @return {Boolean}
35491      */
35492     isLoading : function(){
35493         return this.loading;  
35494     },
35495     
35496     loadComplete : function(deep, anim, callback){
35497         this.loading = false;
35498         this.loaded = true;
35499         this.ui.afterLoad(this);
35500         this.fireEvent("load", this);
35501         this.expand(deep, anim, callback);
35502     },
35503     
35504     /**
35505      * Returns true if this node has been loaded
35506      * @return {Boolean}
35507      */
35508     isLoaded : function(){
35509         return this.loaded;
35510     },
35511     
35512     hasChildNodes : function(){
35513         if(!this.isLeaf() && !this.loaded){
35514             return true;
35515         }else{
35516             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35517         }
35518     },
35519
35520     /**
35521      * Trigger a reload for this node
35522      * @param {Function} callback
35523      */
35524     reload : function(callback){
35525         this.collapse(false, false);
35526         while(this.firstChild){
35527             this.removeChild(this.firstChild);
35528         }
35529         this.childrenRendered = false;
35530         this.loaded = false;
35531         if(this.isHiddenRoot()){
35532             this.expanded = false;
35533         }
35534         this.expand(false, false, callback);
35535     }
35536 });/*
35537  * Based on:
35538  * Ext JS Library 1.1.1
35539  * Copyright(c) 2006-2007, Ext JS, LLC.
35540  *
35541  * Originally Released Under LGPL - original licence link has changed is not relivant.
35542  *
35543  * Fork - LGPL
35544  * <script type="text/javascript">
35545  */
35546  
35547 /**
35548  * @class Roo.tree.TreeNodeUI
35549  * @constructor
35550  * @param {Object} node The node to render
35551  * The TreeNode UI implementation is separate from the
35552  * tree implementation. Unless you are customizing the tree UI,
35553  * you should never have to use this directly.
35554  */
35555 Roo.tree.TreeNodeUI = function(node){
35556     this.node = node;
35557     this.rendered = false;
35558     this.animating = false;
35559     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35560 };
35561
35562 Roo.tree.TreeNodeUI.prototype = {
35563     removeChild : function(node){
35564         if(this.rendered){
35565             this.ctNode.removeChild(node.ui.getEl());
35566         }
35567     },
35568
35569     beforeLoad : function(){
35570          this.addClass("x-tree-node-loading");
35571     },
35572
35573     afterLoad : function(){
35574          this.removeClass("x-tree-node-loading");
35575     },
35576
35577     onTextChange : function(node, text, oldText){
35578         if(this.rendered){
35579             this.textNode.innerHTML = text;
35580         }
35581     },
35582
35583     onDisableChange : function(node, state){
35584         this.disabled = state;
35585         if(state){
35586             this.addClass("x-tree-node-disabled");
35587         }else{
35588             this.removeClass("x-tree-node-disabled");
35589         }
35590     },
35591
35592     onSelectedChange : function(state){
35593         if(state){
35594             this.focus();
35595             this.addClass("x-tree-selected");
35596         }else{
35597             //this.blur();
35598             this.removeClass("x-tree-selected");
35599         }
35600     },
35601
35602     onMove : function(tree, node, oldParent, newParent, index, refNode){
35603         this.childIndent = null;
35604         if(this.rendered){
35605             var targetNode = newParent.ui.getContainer();
35606             if(!targetNode){//target not rendered
35607                 this.holder = document.createElement("div");
35608                 this.holder.appendChild(this.wrap);
35609                 return;
35610             }
35611             var insertBefore = refNode ? refNode.ui.getEl() : null;
35612             if(insertBefore){
35613                 targetNode.insertBefore(this.wrap, insertBefore);
35614             }else{
35615                 targetNode.appendChild(this.wrap);
35616             }
35617             this.node.renderIndent(true);
35618         }
35619     },
35620
35621     addClass : function(cls){
35622         if(this.elNode){
35623             Roo.fly(this.elNode).addClass(cls);
35624         }
35625     },
35626
35627     removeClass : function(cls){
35628         if(this.elNode){
35629             Roo.fly(this.elNode).removeClass(cls);
35630         }
35631     },
35632
35633     remove : function(){
35634         if(this.rendered){
35635             this.holder = document.createElement("div");
35636             this.holder.appendChild(this.wrap);
35637         }
35638     },
35639
35640     fireEvent : function(){
35641         return this.node.fireEvent.apply(this.node, arguments);
35642     },
35643
35644     initEvents : function(){
35645         this.node.on("move", this.onMove, this);
35646         var E = Roo.EventManager;
35647         var a = this.anchor;
35648
35649         var el = Roo.fly(a, '_treeui');
35650
35651         if(Roo.isOpera){ // opera render bug ignores the CSS
35652             el.setStyle("text-decoration", "none");
35653         }
35654
35655         el.on("click", this.onClick, this);
35656         el.on("dblclick", this.onDblClick, this);
35657
35658         if(this.checkbox){
35659             Roo.EventManager.on(this.checkbox,
35660                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35661         }
35662
35663         el.on("contextmenu", this.onContextMenu, this);
35664
35665         var icon = Roo.fly(this.iconNode);
35666         icon.on("click", this.onClick, this);
35667         icon.on("dblclick", this.onDblClick, this);
35668         icon.on("contextmenu", this.onContextMenu, this);
35669         E.on(this.ecNode, "click", this.ecClick, this, true);
35670
35671         if(this.node.disabled){
35672             this.addClass("x-tree-node-disabled");
35673         }
35674         if(this.node.hidden){
35675             this.addClass("x-tree-node-disabled");
35676         }
35677         var ot = this.node.getOwnerTree();
35678         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35679         if(dd && (!this.node.isRoot || ot.rootVisible)){
35680             Roo.dd.Registry.register(this.elNode, {
35681                 node: this.node,
35682                 handles: this.getDDHandles(),
35683                 isHandle: false
35684             });
35685         }
35686     },
35687
35688     getDDHandles : function(){
35689         return [this.iconNode, this.textNode];
35690     },
35691
35692     hide : function(){
35693         if(this.rendered){
35694             this.wrap.style.display = "none";
35695         }
35696     },
35697
35698     show : function(){
35699         if(this.rendered){
35700             this.wrap.style.display = "";
35701         }
35702     },
35703
35704     onContextMenu : function(e){
35705         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35706             e.preventDefault();
35707             this.focus();
35708             this.fireEvent("contextmenu", this.node, e);
35709         }
35710     },
35711
35712     onClick : function(e){
35713         if(this.dropping){
35714             e.stopEvent();
35715             return;
35716         }
35717         if(this.fireEvent("beforeclick", this.node, e) !== false){
35718             if(!this.disabled && this.node.attributes.href){
35719                 this.fireEvent("click", this.node, e);
35720                 return;
35721             }
35722             e.preventDefault();
35723             if(this.disabled){
35724                 return;
35725             }
35726
35727             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35728                 this.node.toggle();
35729             }
35730
35731             this.fireEvent("click", this.node, e);
35732         }else{
35733             e.stopEvent();
35734         }
35735     },
35736
35737     onDblClick : function(e){
35738         e.preventDefault();
35739         if(this.disabled){
35740             return;
35741         }
35742         if(this.checkbox){
35743             this.toggleCheck();
35744         }
35745         if(!this.animating && this.node.hasChildNodes()){
35746             this.node.toggle();
35747         }
35748         this.fireEvent("dblclick", this.node, e);
35749     },
35750
35751     onCheckChange : function(){
35752         var checked = this.checkbox.checked;
35753         this.node.attributes.checked = checked;
35754         this.fireEvent('checkchange', this.node, checked);
35755     },
35756
35757     ecClick : function(e){
35758         if(!this.animating && this.node.hasChildNodes()){
35759             this.node.toggle();
35760         }
35761     },
35762
35763     startDrop : function(){
35764         this.dropping = true;
35765     },
35766
35767     // delayed drop so the click event doesn't get fired on a drop
35768     endDrop : function(){
35769        setTimeout(function(){
35770            this.dropping = false;
35771        }.createDelegate(this), 50);
35772     },
35773
35774     expand : function(){
35775         this.updateExpandIcon();
35776         this.ctNode.style.display = "";
35777     },
35778
35779     focus : function(){
35780         if(!this.node.preventHScroll){
35781             try{this.anchor.focus();
35782             }catch(e){}
35783         }else if(!Roo.isIE){
35784             try{
35785                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35786                 var l = noscroll.scrollLeft;
35787                 this.anchor.focus();
35788                 noscroll.scrollLeft = l;
35789             }catch(e){}
35790         }
35791     },
35792
35793     toggleCheck : function(value){
35794         var cb = this.checkbox;
35795         if(cb){
35796             cb.checked = (value === undefined ? !cb.checked : value);
35797         }
35798     },
35799
35800     blur : function(){
35801         try{
35802             this.anchor.blur();
35803         }catch(e){}
35804     },
35805
35806     animExpand : function(callback){
35807         var ct = Roo.get(this.ctNode);
35808         ct.stopFx();
35809         if(!this.node.hasChildNodes()){
35810             this.updateExpandIcon();
35811             this.ctNode.style.display = "";
35812             Roo.callback(callback);
35813             return;
35814         }
35815         this.animating = true;
35816         this.updateExpandIcon();
35817
35818         ct.slideIn('t', {
35819            callback : function(){
35820                this.animating = false;
35821                Roo.callback(callback);
35822             },
35823             scope: this,
35824             duration: this.node.ownerTree.duration || .25
35825         });
35826     },
35827
35828     highlight : function(){
35829         var tree = this.node.getOwnerTree();
35830         Roo.fly(this.wrap).highlight(
35831             tree.hlColor || "C3DAF9",
35832             {endColor: tree.hlBaseColor}
35833         );
35834     },
35835
35836     collapse : function(){
35837         this.updateExpandIcon();
35838         this.ctNode.style.display = "none";
35839     },
35840
35841     animCollapse : function(callback){
35842         var ct = Roo.get(this.ctNode);
35843         ct.enableDisplayMode('block');
35844         ct.stopFx();
35845
35846         this.animating = true;
35847         this.updateExpandIcon();
35848
35849         ct.slideOut('t', {
35850             callback : function(){
35851                this.animating = false;
35852                Roo.callback(callback);
35853             },
35854             scope: this,
35855             duration: this.node.ownerTree.duration || .25
35856         });
35857     },
35858
35859     getContainer : function(){
35860         return this.ctNode;
35861     },
35862
35863     getEl : function(){
35864         return this.wrap;
35865     },
35866
35867     appendDDGhost : function(ghostNode){
35868         ghostNode.appendChild(this.elNode.cloneNode(true));
35869     },
35870
35871     getDDRepairXY : function(){
35872         return Roo.lib.Dom.getXY(this.iconNode);
35873     },
35874
35875     onRender : function(){
35876         this.render();
35877     },
35878
35879     render : function(bulkRender){
35880         var n = this.node, a = n.attributes;
35881         var targetNode = n.parentNode ?
35882               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35883
35884         if(!this.rendered){
35885             this.rendered = true;
35886
35887             this.renderElements(n, a, targetNode, bulkRender);
35888
35889             if(a.qtip){
35890                if(this.textNode.setAttributeNS){
35891                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35892                    if(a.qtipTitle){
35893                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35894                    }
35895                }else{
35896                    this.textNode.setAttribute("ext:qtip", a.qtip);
35897                    if(a.qtipTitle){
35898                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35899                    }
35900                }
35901             }else if(a.qtipCfg){
35902                 a.qtipCfg.target = Roo.id(this.textNode);
35903                 Roo.QuickTips.register(a.qtipCfg);
35904             }
35905             this.initEvents();
35906             if(!this.node.expanded){
35907                 this.updateExpandIcon();
35908             }
35909         }else{
35910             if(bulkRender === true) {
35911                 targetNode.appendChild(this.wrap);
35912             }
35913         }
35914     },
35915
35916     renderElements : function(n, a, targetNode, bulkRender)
35917     {
35918         // add some indent caching, this helps performance when rendering a large tree
35919         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35920         var t = n.getOwnerTree();
35921         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35922         if (typeof(n.attributes.html) != 'undefined') {
35923             txt = n.attributes.html;
35924         }
35925         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35926         var cb = typeof a.checked == 'boolean';
35927         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35928         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35929             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35930             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35931             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35932             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35933             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35934              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35935                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35936             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35937             "</li>"];
35938
35939         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35940             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35941                                 n.nextSibling.ui.getEl(), buf.join(""));
35942         }else{
35943             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35944         }
35945
35946         this.elNode = this.wrap.childNodes[0];
35947         this.ctNode = this.wrap.childNodes[1];
35948         var cs = this.elNode.childNodes;
35949         this.indentNode = cs[0];
35950         this.ecNode = cs[1];
35951         this.iconNode = cs[2];
35952         var index = 3;
35953         if(cb){
35954             this.checkbox = cs[3];
35955             index++;
35956         }
35957         this.anchor = cs[index];
35958         this.textNode = cs[index].firstChild;
35959     },
35960
35961     getAnchor : function(){
35962         return this.anchor;
35963     },
35964
35965     getTextEl : function(){
35966         return this.textNode;
35967     },
35968
35969     getIconEl : function(){
35970         return this.iconNode;
35971     },
35972
35973     isChecked : function(){
35974         return this.checkbox ? this.checkbox.checked : false;
35975     },
35976
35977     updateExpandIcon : function(){
35978         if(this.rendered){
35979             var n = this.node, c1, c2;
35980             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35981             var hasChild = n.hasChildNodes();
35982             if(hasChild){
35983                 if(n.expanded){
35984                     cls += "-minus";
35985                     c1 = "x-tree-node-collapsed";
35986                     c2 = "x-tree-node-expanded";
35987                 }else{
35988                     cls += "-plus";
35989                     c1 = "x-tree-node-expanded";
35990                     c2 = "x-tree-node-collapsed";
35991                 }
35992                 if(this.wasLeaf){
35993                     this.removeClass("x-tree-node-leaf");
35994                     this.wasLeaf = false;
35995                 }
35996                 if(this.c1 != c1 || this.c2 != c2){
35997                     Roo.fly(this.elNode).replaceClass(c1, c2);
35998                     this.c1 = c1; this.c2 = c2;
35999                 }
36000             }else{
36001                 // this changes non-leafs into leafs if they have no children.
36002                 // it's not very rational behaviour..
36003                 
36004                 if(!this.wasLeaf && this.node.leaf){
36005                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
36006                     delete this.c1;
36007                     delete this.c2;
36008                     this.wasLeaf = true;
36009                 }
36010             }
36011             var ecc = "x-tree-ec-icon "+cls;
36012             if(this.ecc != ecc){
36013                 this.ecNode.className = ecc;
36014                 this.ecc = ecc;
36015             }
36016         }
36017     },
36018
36019     getChildIndent : function(){
36020         if(!this.childIndent){
36021             var buf = [];
36022             var p = this.node;
36023             while(p){
36024                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
36025                     if(!p.isLast()) {
36026                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
36027                     } else {
36028                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
36029                     }
36030                 }
36031                 p = p.parentNode;
36032             }
36033             this.childIndent = buf.join("");
36034         }
36035         return this.childIndent;
36036     },
36037
36038     renderIndent : function(){
36039         if(this.rendered){
36040             var indent = "";
36041             var p = this.node.parentNode;
36042             if(p){
36043                 indent = p.ui.getChildIndent();
36044             }
36045             if(this.indentMarkup != indent){ // don't rerender if not required
36046                 this.indentNode.innerHTML = indent;
36047                 this.indentMarkup = indent;
36048             }
36049             this.updateExpandIcon();
36050         }
36051     }
36052 };
36053
36054 Roo.tree.RootTreeNodeUI = function(){
36055     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36056 };
36057 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36058     render : function(){
36059         if(!this.rendered){
36060             var targetNode = this.node.ownerTree.innerCt.dom;
36061             this.node.expanded = true;
36062             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36063             this.wrap = this.ctNode = targetNode.firstChild;
36064         }
36065     },
36066     collapse : function(){
36067     },
36068     expand : function(){
36069     }
36070 });/*
36071  * Based on:
36072  * Ext JS Library 1.1.1
36073  * Copyright(c) 2006-2007, Ext JS, LLC.
36074  *
36075  * Originally Released Under LGPL - original licence link has changed is not relivant.
36076  *
36077  * Fork - LGPL
36078  * <script type="text/javascript">
36079  */
36080 /**
36081  * @class Roo.tree.TreeLoader
36082  * @extends Roo.util.Observable
36083  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36084  * nodes from a specified URL. The response must be a javascript Array definition
36085  * who's elements are node definition objects. eg:
36086  * <pre><code>
36087 {  success : true,
36088    data :      [
36089    
36090     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36091     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36092     ]
36093 }
36094
36095
36096 </code></pre>
36097  * <br><br>
36098  * The old style respose with just an array is still supported, but not recommended.
36099  * <br><br>
36100  *
36101  * A server request is sent, and child nodes are loaded only when a node is expanded.
36102  * The loading node's id is passed to the server under the parameter name "node" to
36103  * enable the server to produce the correct child nodes.
36104  * <br><br>
36105  * To pass extra parameters, an event handler may be attached to the "beforeload"
36106  * event, and the parameters specified in the TreeLoader's baseParams property:
36107  * <pre><code>
36108     myTreeLoader.on("beforeload", function(treeLoader, node) {
36109         this.baseParams.category = node.attributes.category;
36110     }, this);
36111     
36112 </code></pre>
36113  *
36114  * This would pass an HTTP parameter called "category" to the server containing
36115  * the value of the Node's "category" attribute.
36116  * @constructor
36117  * Creates a new Treeloader.
36118  * @param {Object} config A config object containing config properties.
36119  */
36120 Roo.tree.TreeLoader = function(config){
36121     this.baseParams = {};
36122     this.requestMethod = "POST";
36123     Roo.apply(this, config);
36124
36125     this.addEvents({
36126     
36127         /**
36128          * @event beforeload
36129          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36130          * @param {Object} This TreeLoader object.
36131          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36132          * @param {Object} callback The callback function specified in the {@link #load} call.
36133          */
36134         beforeload : true,
36135         /**
36136          * @event load
36137          * Fires when the node has been successfuly loaded.
36138          * @param {Object} This TreeLoader object.
36139          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36140          * @param {Object} response The response object containing the data from the server.
36141          */
36142         load : true,
36143         /**
36144          * @event loadexception
36145          * Fires if the network request failed.
36146          * @param {Object} This TreeLoader object.
36147          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36148          * @param {Object} response The response object containing the data from the server.
36149          */
36150         loadexception : true,
36151         /**
36152          * @event create
36153          * Fires before a node is created, enabling you to return custom Node types 
36154          * @param {Object} This TreeLoader object.
36155          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36156          */
36157         create : true
36158     });
36159
36160     Roo.tree.TreeLoader.superclass.constructor.call(this);
36161 };
36162
36163 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36164     /**
36165     * @cfg {String} dataUrl The URL from which to request a Json string which
36166     * specifies an array of node definition object representing the child nodes
36167     * to be loaded.
36168     */
36169     /**
36170     * @cfg {String} requestMethod either GET or POST
36171     * defaults to POST (due to BC)
36172     * to be loaded.
36173     */
36174     /**
36175     * @cfg {Object} baseParams (optional) An object containing properties which
36176     * specify HTTP parameters to be passed to each request for child nodes.
36177     */
36178     /**
36179     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36180     * created by this loader. If the attributes sent by the server have an attribute in this object,
36181     * they take priority.
36182     */
36183     /**
36184     * @cfg {Object} uiProviders (optional) An object containing properties which
36185     * 
36186     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36187     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36188     * <i>uiProvider</i> attribute of a returned child node is a string rather
36189     * than a reference to a TreeNodeUI implementation, this that string value
36190     * is used as a property name in the uiProviders object. You can define the provider named
36191     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36192     */
36193     uiProviders : {},
36194
36195     /**
36196     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36197     * child nodes before loading.
36198     */
36199     clearOnLoad : true,
36200
36201     /**
36202     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36203     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36204     * Grid query { data : [ .....] }
36205     */
36206     
36207     root : false,
36208      /**
36209     * @cfg {String} queryParam (optional) 
36210     * Name of the query as it will be passed on the querystring (defaults to 'node')
36211     * eg. the request will be ?node=[id]
36212     */
36213     
36214     
36215     queryParam: false,
36216     
36217     /**
36218      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36219      * This is called automatically when a node is expanded, but may be used to reload
36220      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36221      * @param {Roo.tree.TreeNode} node
36222      * @param {Function} callback
36223      */
36224     load : function(node, callback){
36225         if(this.clearOnLoad){
36226             while(node.firstChild){
36227                 node.removeChild(node.firstChild);
36228             }
36229         }
36230         if(node.attributes.children){ // preloaded json children
36231             var cs = node.attributes.children;
36232             for(var i = 0, len = cs.length; i < len; i++){
36233                 node.appendChild(this.createNode(cs[i]));
36234             }
36235             if(typeof callback == "function"){
36236                 callback();
36237             }
36238         }else if(this.dataUrl){
36239             this.requestData(node, callback);
36240         }
36241     },
36242
36243     getParams: function(node){
36244         var buf = [], bp = this.baseParams;
36245         for(var key in bp){
36246             if(typeof bp[key] != "function"){
36247                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36248             }
36249         }
36250         var n = this.queryParam === false ? 'node' : this.queryParam;
36251         buf.push(n + "=", encodeURIComponent(node.id));
36252         return buf.join("");
36253     },
36254
36255     requestData : function(node, callback){
36256         if(this.fireEvent("beforeload", this, node, callback) !== false){
36257             this.transId = Roo.Ajax.request({
36258                 method:this.requestMethod,
36259                 url: this.dataUrl||this.url,
36260                 success: this.handleResponse,
36261                 failure: this.handleFailure,
36262                 scope: this,
36263                 argument: {callback: callback, node: node},
36264                 params: this.getParams(node)
36265             });
36266         }else{
36267             // if the load is cancelled, make sure we notify
36268             // the node that we are done
36269             if(typeof callback == "function"){
36270                 callback();
36271             }
36272         }
36273     },
36274
36275     isLoading : function(){
36276         return this.transId ? true : false;
36277     },
36278
36279     abort : function(){
36280         if(this.isLoading()){
36281             Roo.Ajax.abort(this.transId);
36282         }
36283     },
36284
36285     // private
36286     createNode : function(attr)
36287     {
36288         // apply baseAttrs, nice idea Corey!
36289         if(this.baseAttrs){
36290             Roo.applyIf(attr, this.baseAttrs);
36291         }
36292         if(this.applyLoader !== false){
36293             attr.loader = this;
36294         }
36295         // uiProvider = depreciated..
36296         
36297         if(typeof(attr.uiProvider) == 'string'){
36298            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36299                 /**  eval:var:attr */ eval(attr.uiProvider);
36300         }
36301         if(typeof(this.uiProviders['default']) != 'undefined') {
36302             attr.uiProvider = this.uiProviders['default'];
36303         }
36304         
36305         this.fireEvent('create', this, attr);
36306         
36307         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36308         return(attr.leaf ?
36309                         new Roo.tree.TreeNode(attr) :
36310                         new Roo.tree.AsyncTreeNode(attr));
36311     },
36312
36313     processResponse : function(response, node, callback)
36314     {
36315         var json = response.responseText;
36316         try {
36317             
36318             var o = Roo.decode(json);
36319             
36320             if (this.root === false && typeof(o.success) != undefined) {
36321                 this.root = 'data'; // the default behaviour for list like data..
36322                 }
36323                 
36324             if (this.root !== false &&  !o.success) {
36325                 // it's a failure condition.
36326                 var a = response.argument;
36327                 this.fireEvent("loadexception", this, a.node, response);
36328                 Roo.log("Load failed - should have a handler really");
36329                 return;
36330             }
36331             
36332             
36333             
36334             if (this.root !== false) {
36335                  o = o[this.root];
36336             }
36337             
36338             for(var i = 0, len = o.length; i < len; i++){
36339                 var n = this.createNode(o[i]);
36340                 if(n){
36341                     node.appendChild(n);
36342                 }
36343             }
36344             if(typeof callback == "function"){
36345                 callback(this, node);
36346             }
36347         }catch(e){
36348             this.handleFailure(response);
36349         }
36350     },
36351
36352     handleResponse : function(response){
36353         this.transId = false;
36354         var a = response.argument;
36355         this.processResponse(response, a.node, a.callback);
36356         this.fireEvent("load", this, a.node, response);
36357     },
36358
36359     handleFailure : function(response)
36360     {
36361         // should handle failure better..
36362         this.transId = false;
36363         var a = response.argument;
36364         this.fireEvent("loadexception", this, a.node, response);
36365         if(typeof a.callback == "function"){
36366             a.callback(this, a.node);
36367         }
36368     }
36369 });/*
36370  * Based on:
36371  * Ext JS Library 1.1.1
36372  * Copyright(c) 2006-2007, Ext JS, LLC.
36373  *
36374  * Originally Released Under LGPL - original licence link has changed is not relivant.
36375  *
36376  * Fork - LGPL
36377  * <script type="text/javascript">
36378  */
36379
36380 /**
36381 * @class Roo.tree.TreeFilter
36382 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36383 * @param {TreePanel} tree
36384 * @param {Object} config (optional)
36385  */
36386 Roo.tree.TreeFilter = function(tree, config){
36387     this.tree = tree;
36388     this.filtered = {};
36389     Roo.apply(this, config);
36390 };
36391
36392 Roo.tree.TreeFilter.prototype = {
36393     clearBlank:false,
36394     reverse:false,
36395     autoClear:false,
36396     remove:false,
36397
36398      /**
36399      * Filter the data by a specific attribute.
36400      * @param {String/RegExp} value Either string that the attribute value
36401      * should start with or a RegExp to test against the attribute
36402      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36403      * @param {TreeNode} startNode (optional) The node to start the filter at.
36404      */
36405     filter : function(value, attr, startNode){
36406         attr = attr || "text";
36407         var f;
36408         if(typeof value == "string"){
36409             var vlen = value.length;
36410             // auto clear empty filter
36411             if(vlen == 0 && this.clearBlank){
36412                 this.clear();
36413                 return;
36414             }
36415             value = value.toLowerCase();
36416             f = function(n){
36417                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36418             };
36419         }else if(value.exec){ // regex?
36420             f = function(n){
36421                 return value.test(n.attributes[attr]);
36422             };
36423         }else{
36424             throw 'Illegal filter type, must be string or regex';
36425         }
36426         this.filterBy(f, null, startNode);
36427         },
36428
36429     /**
36430      * Filter by a function. The passed function will be called with each
36431      * node in the tree (or from the startNode). If the function returns true, the node is kept
36432      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36433      * @param {Function} fn The filter function
36434      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36435      */
36436     filterBy : function(fn, scope, startNode){
36437         startNode = startNode || this.tree.root;
36438         if(this.autoClear){
36439             this.clear();
36440         }
36441         var af = this.filtered, rv = this.reverse;
36442         var f = function(n){
36443             if(n == startNode){
36444                 return true;
36445             }
36446             if(af[n.id]){
36447                 return false;
36448             }
36449             var m = fn.call(scope || n, n);
36450             if(!m || rv){
36451                 af[n.id] = n;
36452                 n.ui.hide();
36453                 return false;
36454             }
36455             return true;
36456         };
36457         startNode.cascade(f);
36458         if(this.remove){
36459            for(var id in af){
36460                if(typeof id != "function"){
36461                    var n = af[id];
36462                    if(n && n.parentNode){
36463                        n.parentNode.removeChild(n);
36464                    }
36465                }
36466            }
36467         }
36468     },
36469
36470     /**
36471      * Clears the current filter. Note: with the "remove" option
36472      * set a filter cannot be cleared.
36473      */
36474     clear : function(){
36475         var t = this.tree;
36476         var af = this.filtered;
36477         for(var id in af){
36478             if(typeof id != "function"){
36479                 var n = af[id];
36480                 if(n){
36481                     n.ui.show();
36482                 }
36483             }
36484         }
36485         this.filtered = {};
36486     }
36487 };
36488 /*
36489  * Based on:
36490  * Ext JS Library 1.1.1
36491  * Copyright(c) 2006-2007, Ext JS, LLC.
36492  *
36493  * Originally Released Under LGPL - original licence link has changed is not relivant.
36494  *
36495  * Fork - LGPL
36496  * <script type="text/javascript">
36497  */
36498  
36499
36500 /**
36501  * @class Roo.tree.TreeSorter
36502  * Provides sorting of nodes in a TreePanel
36503  * 
36504  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36505  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36506  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36507  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36508  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36509  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36510  * @constructor
36511  * @param {TreePanel} tree
36512  * @param {Object} config
36513  */
36514 Roo.tree.TreeSorter = function(tree, config){
36515     Roo.apply(this, config);
36516     tree.on("beforechildrenrendered", this.doSort, this);
36517     tree.on("append", this.updateSort, this);
36518     tree.on("insert", this.updateSort, this);
36519     
36520     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36521     var p = this.property || "text";
36522     var sortType = this.sortType;
36523     var fs = this.folderSort;
36524     var cs = this.caseSensitive === true;
36525     var leafAttr = this.leafAttr || 'leaf';
36526
36527     this.sortFn = function(n1, n2){
36528         if(fs){
36529             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36530                 return 1;
36531             }
36532             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36533                 return -1;
36534             }
36535         }
36536         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36537         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36538         if(v1 < v2){
36539                         return dsc ? +1 : -1;
36540                 }else if(v1 > v2){
36541                         return dsc ? -1 : +1;
36542         }else{
36543                 return 0;
36544         }
36545     };
36546 };
36547
36548 Roo.tree.TreeSorter.prototype = {
36549     doSort : function(node){
36550         node.sort(this.sortFn);
36551     },
36552     
36553     compareNodes : function(n1, n2){
36554         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36555     },
36556     
36557     updateSort : function(tree, node){
36558         if(node.childrenRendered){
36559             this.doSort.defer(1, this, [node]);
36560         }
36561     }
36562 };/*
36563  * Based on:
36564  * Ext JS Library 1.1.1
36565  * Copyright(c) 2006-2007, Ext JS, LLC.
36566  *
36567  * Originally Released Under LGPL - original licence link has changed is not relivant.
36568  *
36569  * Fork - LGPL
36570  * <script type="text/javascript">
36571  */
36572
36573 if(Roo.dd.DropZone){
36574     
36575 Roo.tree.TreeDropZone = function(tree, config){
36576     this.allowParentInsert = false;
36577     this.allowContainerDrop = false;
36578     this.appendOnly = false;
36579     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36580     this.tree = tree;
36581     this.lastInsertClass = "x-tree-no-status";
36582     this.dragOverData = {};
36583 };
36584
36585 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36586     ddGroup : "TreeDD",
36587     scroll:  true,
36588     
36589     expandDelay : 1000,
36590     
36591     expandNode : function(node){
36592         if(node.hasChildNodes() && !node.isExpanded()){
36593             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36594         }
36595     },
36596     
36597     queueExpand : function(node){
36598         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36599     },
36600     
36601     cancelExpand : function(){
36602         if(this.expandProcId){
36603             clearTimeout(this.expandProcId);
36604             this.expandProcId = false;
36605         }
36606     },
36607     
36608     isValidDropPoint : function(n, pt, dd, e, data){
36609         if(!n || !data){ return false; }
36610         var targetNode = n.node;
36611         var dropNode = data.node;
36612         // default drop rules
36613         if(!(targetNode && targetNode.isTarget && pt)){
36614             return false;
36615         }
36616         if(pt == "append" && targetNode.allowChildren === false){
36617             return false;
36618         }
36619         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36620             return false;
36621         }
36622         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36623             return false;
36624         }
36625         // reuse the object
36626         var overEvent = this.dragOverData;
36627         overEvent.tree = this.tree;
36628         overEvent.target = targetNode;
36629         overEvent.data = data;
36630         overEvent.point = pt;
36631         overEvent.source = dd;
36632         overEvent.rawEvent = e;
36633         overEvent.dropNode = dropNode;
36634         overEvent.cancel = false;  
36635         var result = this.tree.fireEvent("nodedragover", overEvent);
36636         return overEvent.cancel === false && result !== false;
36637     },
36638     
36639     getDropPoint : function(e, n, dd)
36640     {
36641         var tn = n.node;
36642         if(tn.isRoot){
36643             return tn.allowChildren !== false ? "append" : false; // always append for root
36644         }
36645         var dragEl = n.ddel;
36646         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36647         var y = Roo.lib.Event.getPageY(e);
36648         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36649         
36650         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36651         var noAppend = tn.allowChildren === false;
36652         if(this.appendOnly || tn.parentNode.allowChildren === false){
36653             return noAppend ? false : "append";
36654         }
36655         var noBelow = false;
36656         if(!this.allowParentInsert){
36657             noBelow = tn.hasChildNodes() && tn.isExpanded();
36658         }
36659         var q = (b - t) / (noAppend ? 2 : 3);
36660         if(y >= t && y < (t + q)){
36661             return "above";
36662         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36663             return "below";
36664         }else{
36665             return "append";
36666         }
36667     },
36668     
36669     onNodeEnter : function(n, dd, e, data)
36670     {
36671         this.cancelExpand();
36672     },
36673     
36674     onNodeOver : function(n, dd, e, data)
36675     {
36676        
36677         var pt = this.getDropPoint(e, n, dd);
36678         var node = n.node;
36679         
36680         // auto node expand check
36681         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36682             this.queueExpand(node);
36683         }else if(pt != "append"){
36684             this.cancelExpand();
36685         }
36686         
36687         // set the insert point style on the target node
36688         var returnCls = this.dropNotAllowed;
36689         if(this.isValidDropPoint(n, pt, dd, e, data)){
36690            if(pt){
36691                var el = n.ddel;
36692                var cls;
36693                if(pt == "above"){
36694                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36695                    cls = "x-tree-drag-insert-above";
36696                }else if(pt == "below"){
36697                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36698                    cls = "x-tree-drag-insert-below";
36699                }else{
36700                    returnCls = "x-tree-drop-ok-append";
36701                    cls = "x-tree-drag-append";
36702                }
36703                if(this.lastInsertClass != cls){
36704                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36705                    this.lastInsertClass = cls;
36706                }
36707            }
36708        }
36709        return returnCls;
36710     },
36711     
36712     onNodeOut : function(n, dd, e, data){
36713         
36714         this.cancelExpand();
36715         this.removeDropIndicators(n);
36716     },
36717     
36718     onNodeDrop : function(n, dd, e, data){
36719         var point = this.getDropPoint(e, n, dd);
36720         var targetNode = n.node;
36721         targetNode.ui.startDrop();
36722         if(!this.isValidDropPoint(n, point, dd, e, data)){
36723             targetNode.ui.endDrop();
36724             return false;
36725         }
36726         // first try to find the drop node
36727         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36728         var dropEvent = {
36729             tree : this.tree,
36730             target: targetNode,
36731             data: data,
36732             point: point,
36733             source: dd,
36734             rawEvent: e,
36735             dropNode: dropNode,
36736             cancel: !dropNode   
36737         };
36738         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36739         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36740             targetNode.ui.endDrop();
36741             return false;
36742         }
36743         // allow target changing
36744         targetNode = dropEvent.target;
36745         if(point == "append" && !targetNode.isExpanded()){
36746             targetNode.expand(false, null, function(){
36747                 this.completeDrop(dropEvent);
36748             }.createDelegate(this));
36749         }else{
36750             this.completeDrop(dropEvent);
36751         }
36752         return true;
36753     },
36754     
36755     completeDrop : function(de){
36756         var ns = de.dropNode, p = de.point, t = de.target;
36757         if(!(ns instanceof Array)){
36758             ns = [ns];
36759         }
36760         var n;
36761         for(var i = 0, len = ns.length; i < len; i++){
36762             n = ns[i];
36763             if(p == "above"){
36764                 t.parentNode.insertBefore(n, t);
36765             }else if(p == "below"){
36766                 t.parentNode.insertBefore(n, t.nextSibling);
36767             }else{
36768                 t.appendChild(n);
36769             }
36770         }
36771         n.ui.focus();
36772         if(this.tree.hlDrop){
36773             n.ui.highlight();
36774         }
36775         t.ui.endDrop();
36776         this.tree.fireEvent("nodedrop", de);
36777     },
36778     
36779     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36780         if(this.tree.hlDrop){
36781             dropNode.ui.focus();
36782             dropNode.ui.highlight();
36783         }
36784         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36785     },
36786     
36787     getTree : function(){
36788         return this.tree;
36789     },
36790     
36791     removeDropIndicators : function(n){
36792         if(n && n.ddel){
36793             var el = n.ddel;
36794             Roo.fly(el).removeClass([
36795                     "x-tree-drag-insert-above",
36796                     "x-tree-drag-insert-below",
36797                     "x-tree-drag-append"]);
36798             this.lastInsertClass = "_noclass";
36799         }
36800     },
36801     
36802     beforeDragDrop : function(target, e, id){
36803         this.cancelExpand();
36804         return true;
36805     },
36806     
36807     afterRepair : function(data){
36808         if(data && Roo.enableFx){
36809             data.node.ui.highlight();
36810         }
36811         this.hideProxy();
36812     } 
36813     
36814 });
36815
36816 }
36817 /*
36818  * Based on:
36819  * Ext JS Library 1.1.1
36820  * Copyright(c) 2006-2007, Ext JS, LLC.
36821  *
36822  * Originally Released Under LGPL - original licence link has changed is not relivant.
36823  *
36824  * Fork - LGPL
36825  * <script type="text/javascript">
36826  */
36827  
36828
36829 if(Roo.dd.DragZone){
36830 Roo.tree.TreeDragZone = function(tree, config){
36831     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36832     this.tree = tree;
36833 };
36834
36835 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36836     ddGroup : "TreeDD",
36837    
36838     onBeforeDrag : function(data, e){
36839         var n = data.node;
36840         return n && n.draggable && !n.disabled;
36841     },
36842      
36843     
36844     onInitDrag : function(e){
36845         var data = this.dragData;
36846         this.tree.getSelectionModel().select(data.node);
36847         this.proxy.update("");
36848         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36849         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36850     },
36851     
36852     getRepairXY : function(e, data){
36853         return data.node.ui.getDDRepairXY();
36854     },
36855     
36856     onEndDrag : function(data, e){
36857         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36858         
36859         
36860     },
36861     
36862     onValidDrop : function(dd, e, id){
36863         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36864         this.hideProxy();
36865     },
36866     
36867     beforeInvalidDrop : function(e, id){
36868         // this scrolls the original position back into view
36869         var sm = this.tree.getSelectionModel();
36870         sm.clearSelections();
36871         sm.select(this.dragData.node);
36872     }
36873 });
36874 }/*
36875  * Based on:
36876  * Ext JS Library 1.1.1
36877  * Copyright(c) 2006-2007, Ext JS, LLC.
36878  *
36879  * Originally Released Under LGPL - original licence link has changed is not relivant.
36880  *
36881  * Fork - LGPL
36882  * <script type="text/javascript">
36883  */
36884 /**
36885  * @class Roo.tree.TreeEditor
36886  * @extends Roo.Editor
36887  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36888  * as the editor field.
36889  * @constructor
36890  * @param {Object} config (used to be the tree panel.)
36891  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36892  * 
36893  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36894  * @cfg {Roo.form.TextField|Object} field The field configuration
36895  *
36896  * 
36897  */
36898 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36899     var tree = config;
36900     var field;
36901     if (oldconfig) { // old style..
36902         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36903     } else {
36904         // new style..
36905         tree = config.tree;
36906         config.field = config.field  || {};
36907         config.field.xtype = 'TextField';
36908         field = Roo.factory(config.field, Roo.form);
36909     }
36910     config = config || {};
36911     
36912     
36913     this.addEvents({
36914         /**
36915          * @event beforenodeedit
36916          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36917          * false from the handler of this event.
36918          * @param {Editor} this
36919          * @param {Roo.tree.Node} node 
36920          */
36921         "beforenodeedit" : true
36922     });
36923     
36924     //Roo.log(config);
36925     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36926
36927     this.tree = tree;
36928
36929     tree.on('beforeclick', this.beforeNodeClick, this);
36930     tree.getTreeEl().on('mousedown', this.hide, this);
36931     this.on('complete', this.updateNode, this);
36932     this.on('beforestartedit', this.fitToTree, this);
36933     this.on('startedit', this.bindScroll, this, {delay:10});
36934     this.on('specialkey', this.onSpecialKey, this);
36935 };
36936
36937 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36938     /**
36939      * @cfg {String} alignment
36940      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36941      */
36942     alignment: "l-l",
36943     // inherit
36944     autoSize: false,
36945     /**
36946      * @cfg {Boolean} hideEl
36947      * True to hide the bound element while the editor is displayed (defaults to false)
36948      */
36949     hideEl : false,
36950     /**
36951      * @cfg {String} cls
36952      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36953      */
36954     cls: "x-small-editor x-tree-editor",
36955     /**
36956      * @cfg {Boolean} shim
36957      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36958      */
36959     shim:false,
36960     // inherit
36961     shadow:"frame",
36962     /**
36963      * @cfg {Number} maxWidth
36964      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36965      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36966      * scroll and client offsets into account prior to each edit.
36967      */
36968     maxWidth: 250,
36969
36970     editDelay : 350,
36971
36972     // private
36973     fitToTree : function(ed, el){
36974         var td = this.tree.getTreeEl().dom, nd = el.dom;
36975         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36976             td.scrollLeft = nd.offsetLeft;
36977         }
36978         var w = Math.min(
36979                 this.maxWidth,
36980                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36981         this.setSize(w, '');
36982         
36983         return this.fireEvent('beforenodeedit', this, this.editNode);
36984         
36985     },
36986
36987     // private
36988     triggerEdit : function(node){
36989         this.completeEdit();
36990         this.editNode = node;
36991         this.startEdit(node.ui.textNode, node.text);
36992     },
36993
36994     // private
36995     bindScroll : function(){
36996         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36997     },
36998
36999     // private
37000     beforeNodeClick : function(node, e){
37001         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
37002         this.lastClick = new Date();
37003         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
37004             e.stopEvent();
37005             this.triggerEdit(node);
37006             return false;
37007         }
37008         return true;
37009     },
37010
37011     // private
37012     updateNode : function(ed, value){
37013         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
37014         this.editNode.setText(value);
37015     },
37016
37017     // private
37018     onHide : function(){
37019         Roo.tree.TreeEditor.superclass.onHide.call(this);
37020         if(this.editNode){
37021             this.editNode.ui.focus();
37022         }
37023     },
37024
37025     // private
37026     onSpecialKey : function(field, e){
37027         var k = e.getKey();
37028         if(k == e.ESC){
37029             e.stopEvent();
37030             this.cancelEdit();
37031         }else if(k == e.ENTER && !e.hasModifier()){
37032             e.stopEvent();
37033             this.completeEdit();
37034         }
37035     }
37036 });//<Script type="text/javascript">
37037 /*
37038  * Based on:
37039  * Ext JS Library 1.1.1
37040  * Copyright(c) 2006-2007, Ext JS, LLC.
37041  *
37042  * Originally Released Under LGPL - original licence link has changed is not relivant.
37043  *
37044  * Fork - LGPL
37045  * <script type="text/javascript">
37046  */
37047  
37048 /**
37049  * Not documented??? - probably should be...
37050  */
37051
37052 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37053     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37054     
37055     renderElements : function(n, a, targetNode, bulkRender){
37056         //consel.log("renderElements?");
37057         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37058
37059         var t = n.getOwnerTree();
37060         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37061         
37062         var cols = t.columns;
37063         var bw = t.borderWidth;
37064         var c = cols[0];
37065         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37066          var cb = typeof a.checked == "boolean";
37067         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37068         var colcls = 'x-t-' + tid + '-c0';
37069         var buf = [
37070             '<li class="x-tree-node">',
37071             
37072                 
37073                 '<div class="x-tree-node-el ', a.cls,'">',
37074                     // extran...
37075                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37076                 
37077                 
37078                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37079                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37080                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37081                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37082                            (a.iconCls ? ' '+a.iconCls : ''),
37083                            '" unselectable="on" />',
37084                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37085                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37086                              
37087                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37088                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37089                             '<span unselectable="on" qtip="' + tx + '">',
37090                              tx,
37091                              '</span></a>' ,
37092                     '</div>',
37093                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37094                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37095                  ];
37096         for(var i = 1, len = cols.length; i < len; i++){
37097             c = cols[i];
37098             colcls = 'x-t-' + tid + '-c' +i;
37099             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37100             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37101                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37102                       "</div>");
37103          }
37104          
37105          buf.push(
37106             '</a>',
37107             '<div class="x-clear"></div></div>',
37108             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37109             "</li>");
37110         
37111         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37112             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37113                                 n.nextSibling.ui.getEl(), buf.join(""));
37114         }else{
37115             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37116         }
37117         var el = this.wrap.firstChild;
37118         this.elRow = el;
37119         this.elNode = el.firstChild;
37120         this.ranchor = el.childNodes[1];
37121         this.ctNode = this.wrap.childNodes[1];
37122         var cs = el.firstChild.childNodes;
37123         this.indentNode = cs[0];
37124         this.ecNode = cs[1];
37125         this.iconNode = cs[2];
37126         var index = 3;
37127         if(cb){
37128             this.checkbox = cs[3];
37129             index++;
37130         }
37131         this.anchor = cs[index];
37132         
37133         this.textNode = cs[index].firstChild;
37134         
37135         //el.on("click", this.onClick, this);
37136         //el.on("dblclick", this.onDblClick, this);
37137         
37138         
37139        // console.log(this);
37140     },
37141     initEvents : function(){
37142         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37143         
37144             
37145         var a = this.ranchor;
37146
37147         var el = Roo.get(a);
37148
37149         if(Roo.isOpera){ // opera render bug ignores the CSS
37150             el.setStyle("text-decoration", "none");
37151         }
37152
37153         el.on("click", this.onClick, this);
37154         el.on("dblclick", this.onDblClick, this);
37155         el.on("contextmenu", this.onContextMenu, this);
37156         
37157     },
37158     
37159     /*onSelectedChange : function(state){
37160         if(state){
37161             this.focus();
37162             this.addClass("x-tree-selected");
37163         }else{
37164             //this.blur();
37165             this.removeClass("x-tree-selected");
37166         }
37167     },*/
37168     addClass : function(cls){
37169         if(this.elRow){
37170             Roo.fly(this.elRow).addClass(cls);
37171         }
37172         
37173     },
37174     
37175     
37176     removeClass : function(cls){
37177         if(this.elRow){
37178             Roo.fly(this.elRow).removeClass(cls);
37179         }
37180     }
37181
37182     
37183     
37184 });//<Script type="text/javascript">
37185
37186 /*
37187  * Based on:
37188  * Ext JS Library 1.1.1
37189  * Copyright(c) 2006-2007, Ext JS, LLC.
37190  *
37191  * Originally Released Under LGPL - original licence link has changed is not relivant.
37192  *
37193  * Fork - LGPL
37194  * <script type="text/javascript">
37195  */
37196  
37197
37198 /**
37199  * @class Roo.tree.ColumnTree
37200  * @extends Roo.data.TreePanel
37201  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37202  * @cfg {int} borderWidth  compined right/left border allowance
37203  * @constructor
37204  * @param {String/HTMLElement/Element} el The container element
37205  * @param {Object} config
37206  */
37207 Roo.tree.ColumnTree =  function(el, config)
37208 {
37209    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37210    this.addEvents({
37211         /**
37212         * @event resize
37213         * Fire this event on a container when it resizes
37214         * @param {int} w Width
37215         * @param {int} h Height
37216         */
37217        "resize" : true
37218     });
37219     this.on('resize', this.onResize, this);
37220 };
37221
37222 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37223     //lines:false,
37224     
37225     
37226     borderWidth: Roo.isBorderBox ? 0 : 2, 
37227     headEls : false,
37228     
37229     render : function(){
37230         // add the header.....
37231        
37232         Roo.tree.ColumnTree.superclass.render.apply(this);
37233         
37234         this.el.addClass('x-column-tree');
37235         
37236         this.headers = this.el.createChild(
37237             {cls:'x-tree-headers'},this.innerCt.dom);
37238    
37239         var cols = this.columns, c;
37240         var totalWidth = 0;
37241         this.headEls = [];
37242         var  len = cols.length;
37243         for(var i = 0; i < len; i++){
37244              c = cols[i];
37245              totalWidth += c.width;
37246             this.headEls.push(this.headers.createChild({
37247                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37248                  cn: {
37249                      cls:'x-tree-hd-text',
37250                      html: c.header
37251                  },
37252                  style:'width:'+(c.width-this.borderWidth)+'px;'
37253              }));
37254         }
37255         this.headers.createChild({cls:'x-clear'});
37256         // prevent floats from wrapping when clipped
37257         this.headers.setWidth(totalWidth);
37258         //this.innerCt.setWidth(totalWidth);
37259         this.innerCt.setStyle({ overflow: 'auto' });
37260         this.onResize(this.width, this.height);
37261              
37262         
37263     },
37264     onResize : function(w,h)
37265     {
37266         this.height = h;
37267         this.width = w;
37268         // resize cols..
37269         this.innerCt.setWidth(this.width);
37270         this.innerCt.setHeight(this.height-20);
37271         
37272         // headers...
37273         var cols = this.columns, c;
37274         var totalWidth = 0;
37275         var expEl = false;
37276         var len = cols.length;
37277         for(var i = 0; i < len; i++){
37278             c = cols[i];
37279             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37280                 // it's the expander..
37281                 expEl  = this.headEls[i];
37282                 continue;
37283             }
37284             totalWidth += c.width;
37285             
37286         }
37287         if (expEl) {
37288             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37289         }
37290         this.headers.setWidth(w-20);
37291
37292         
37293         
37294         
37295     }
37296 });
37297 /*
37298  * Based on:
37299  * Ext JS Library 1.1.1
37300  * Copyright(c) 2006-2007, Ext JS, LLC.
37301  *
37302  * Originally Released Under LGPL - original licence link has changed is not relivant.
37303  *
37304  * Fork - LGPL
37305  * <script type="text/javascript">
37306  */
37307  
37308 /**
37309  * @class Roo.menu.Menu
37310  * @extends Roo.util.Observable
37311  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37312  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37313  * @constructor
37314  * Creates a new Menu
37315  * @param {Object} config Configuration options
37316  */
37317 Roo.menu.Menu = function(config){
37318     
37319     Roo.menu.Menu.superclass.constructor.call(this, config);
37320     
37321     this.id = this.id || Roo.id();
37322     this.addEvents({
37323         /**
37324          * @event beforeshow
37325          * Fires before this menu is displayed
37326          * @param {Roo.menu.Menu} this
37327          */
37328         beforeshow : true,
37329         /**
37330          * @event beforehide
37331          * Fires before this menu is hidden
37332          * @param {Roo.menu.Menu} this
37333          */
37334         beforehide : true,
37335         /**
37336          * @event show
37337          * Fires after this menu is displayed
37338          * @param {Roo.menu.Menu} this
37339          */
37340         show : true,
37341         /**
37342          * @event hide
37343          * Fires after this menu is hidden
37344          * @param {Roo.menu.Menu} this
37345          */
37346         hide : true,
37347         /**
37348          * @event click
37349          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37350          * @param {Roo.menu.Menu} this
37351          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37352          * @param {Roo.EventObject} e
37353          */
37354         click : true,
37355         /**
37356          * @event mouseover
37357          * Fires when the mouse is hovering over this menu
37358          * @param {Roo.menu.Menu} this
37359          * @param {Roo.EventObject} e
37360          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37361          */
37362         mouseover : true,
37363         /**
37364          * @event mouseout
37365          * Fires when the mouse exits this menu
37366          * @param {Roo.menu.Menu} this
37367          * @param {Roo.EventObject} e
37368          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37369          */
37370         mouseout : true,
37371         /**
37372          * @event itemclick
37373          * Fires when a menu item contained in this menu is clicked
37374          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37375          * @param {Roo.EventObject} e
37376          */
37377         itemclick: true
37378     });
37379     if (this.registerMenu) {
37380         Roo.menu.MenuMgr.register(this);
37381     }
37382     
37383     var mis = this.items;
37384     this.items = new Roo.util.MixedCollection();
37385     if(mis){
37386         this.add.apply(this, mis);
37387     }
37388 };
37389
37390 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37391     /**
37392      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37393      */
37394     minWidth : 120,
37395     /**
37396      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37397      * for bottom-right shadow (defaults to "sides")
37398      */
37399     shadow : "sides",
37400     /**
37401      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37402      * this menu (defaults to "tl-tr?")
37403      */
37404     subMenuAlign : "tl-tr?",
37405     /**
37406      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37407      * relative to its element of origin (defaults to "tl-bl?")
37408      */
37409     defaultAlign : "tl-bl?",
37410     /**
37411      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37412      */
37413     allowOtherMenus : false,
37414     /**
37415      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37416      */
37417     registerMenu : true,
37418
37419     hidden:true,
37420
37421     // private
37422     render : function(){
37423         if(this.el){
37424             return;
37425         }
37426         var el = this.el = new Roo.Layer({
37427             cls: "x-menu",
37428             shadow:this.shadow,
37429             constrain: false,
37430             parentEl: this.parentEl || document.body,
37431             zindex:15000
37432         });
37433
37434         this.keyNav = new Roo.menu.MenuNav(this);
37435
37436         if(this.plain){
37437             el.addClass("x-menu-plain");
37438         }
37439         if(this.cls){
37440             el.addClass(this.cls);
37441         }
37442         // generic focus element
37443         this.focusEl = el.createChild({
37444             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37445         });
37446         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37447         //disabling touch- as it's causing issues ..
37448         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37449         ul.on('click'   , this.onClick, this);
37450         
37451         
37452         ul.on("mouseover", this.onMouseOver, this);
37453         ul.on("mouseout", this.onMouseOut, this);
37454         this.items.each(function(item){
37455             if (item.hidden) {
37456                 return;
37457             }
37458             
37459             var li = document.createElement("li");
37460             li.className = "x-menu-list-item";
37461             ul.dom.appendChild(li);
37462             item.render(li, this);
37463         }, this);
37464         this.ul = ul;
37465         this.autoWidth();
37466     },
37467
37468     // private
37469     autoWidth : function(){
37470         var el = this.el, ul = this.ul;
37471         if(!el){
37472             return;
37473         }
37474         var w = this.width;
37475         if(w){
37476             el.setWidth(w);
37477         }else if(Roo.isIE){
37478             el.setWidth(this.minWidth);
37479             var t = el.dom.offsetWidth; // force recalc
37480             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37481         }
37482     },
37483
37484     // private
37485     delayAutoWidth : function(){
37486         if(this.rendered){
37487             if(!this.awTask){
37488                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37489             }
37490             this.awTask.delay(20);
37491         }
37492     },
37493
37494     // private
37495     findTargetItem : function(e){
37496         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37497         if(t && t.menuItemId){
37498             return this.items.get(t.menuItemId);
37499         }
37500     },
37501
37502     // private
37503     onClick : function(e){
37504         Roo.log("menu.onClick");
37505         var t = this.findTargetItem(e);
37506         if(!t){
37507             return;
37508         }
37509         Roo.log(e);
37510         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37511             if(t == this.activeItem && t.shouldDeactivate(e)){
37512                 this.activeItem.deactivate();
37513                 delete this.activeItem;
37514                 return;
37515             }
37516             if(t.canActivate){
37517                 this.setActiveItem(t, true);
37518             }
37519             return;
37520             
37521             
37522         }
37523         
37524         t.onClick(e);
37525         this.fireEvent("click", this, t, e);
37526     },
37527
37528     // private
37529     setActiveItem : function(item, autoExpand){
37530         if(item != this.activeItem){
37531             if(this.activeItem){
37532                 this.activeItem.deactivate();
37533             }
37534             this.activeItem = item;
37535             item.activate(autoExpand);
37536         }else if(autoExpand){
37537             item.expandMenu();
37538         }
37539     },
37540
37541     // private
37542     tryActivate : function(start, step){
37543         var items = this.items;
37544         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37545             var item = items.get(i);
37546             if(!item.disabled && item.canActivate){
37547                 this.setActiveItem(item, false);
37548                 return item;
37549             }
37550         }
37551         return false;
37552     },
37553
37554     // private
37555     onMouseOver : function(e){
37556         var t;
37557         if(t = this.findTargetItem(e)){
37558             if(t.canActivate && !t.disabled){
37559                 this.setActiveItem(t, true);
37560             }
37561         }
37562         this.fireEvent("mouseover", this, e, t);
37563     },
37564
37565     // private
37566     onMouseOut : function(e){
37567         var t;
37568         if(t = this.findTargetItem(e)){
37569             if(t == this.activeItem && t.shouldDeactivate(e)){
37570                 this.activeItem.deactivate();
37571                 delete this.activeItem;
37572             }
37573         }
37574         this.fireEvent("mouseout", this, e, t);
37575     },
37576
37577     /**
37578      * Read-only.  Returns true if the menu is currently displayed, else false.
37579      * @type Boolean
37580      */
37581     isVisible : function(){
37582         return this.el && !this.hidden;
37583     },
37584
37585     /**
37586      * Displays this menu relative to another element
37587      * @param {String/HTMLElement/Roo.Element} element The element to align to
37588      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37589      * the element (defaults to this.defaultAlign)
37590      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37591      */
37592     show : function(el, pos, parentMenu){
37593         this.parentMenu = parentMenu;
37594         if(!this.el){
37595             this.render();
37596         }
37597         this.fireEvent("beforeshow", this);
37598         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37599     },
37600
37601     /**
37602      * Displays this menu at a specific xy position
37603      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37604      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37605      */
37606     showAt : function(xy, parentMenu, /* private: */_e){
37607         this.parentMenu = parentMenu;
37608         if(!this.el){
37609             this.render();
37610         }
37611         if(_e !== false){
37612             this.fireEvent("beforeshow", this);
37613             xy = this.el.adjustForConstraints(xy);
37614         }
37615         this.el.setXY(xy);
37616         this.el.show();
37617         this.hidden = false;
37618         this.focus();
37619         this.fireEvent("show", this);
37620     },
37621
37622     focus : function(){
37623         if(!this.hidden){
37624             this.doFocus.defer(50, this);
37625         }
37626     },
37627
37628     doFocus : function(){
37629         if(!this.hidden){
37630             this.focusEl.focus();
37631         }
37632     },
37633
37634     /**
37635      * Hides this menu and optionally all parent menus
37636      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37637      */
37638     hide : function(deep){
37639         if(this.el && this.isVisible()){
37640             this.fireEvent("beforehide", this);
37641             if(this.activeItem){
37642                 this.activeItem.deactivate();
37643                 this.activeItem = null;
37644             }
37645             this.el.hide();
37646             this.hidden = true;
37647             this.fireEvent("hide", this);
37648         }
37649         if(deep === true && this.parentMenu){
37650             this.parentMenu.hide(true);
37651         }
37652     },
37653
37654     /**
37655      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37656      * Any of the following are valid:
37657      * <ul>
37658      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37659      * <li>An HTMLElement object which will be converted to a menu item</li>
37660      * <li>A menu item config object that will be created as a new menu item</li>
37661      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37662      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37663      * </ul>
37664      * Usage:
37665      * <pre><code>
37666 // Create the menu
37667 var menu = new Roo.menu.Menu();
37668
37669 // Create a menu item to add by reference
37670 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37671
37672 // Add a bunch of items at once using different methods.
37673 // Only the last item added will be returned.
37674 var item = menu.add(
37675     menuItem,                // add existing item by ref
37676     'Dynamic Item',          // new TextItem
37677     '-',                     // new separator
37678     { text: 'Config Item' }  // new item by config
37679 );
37680 </code></pre>
37681      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37682      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37683      */
37684     add : function(){
37685         var a = arguments, l = a.length, item;
37686         for(var i = 0; i < l; i++){
37687             var el = a[i];
37688             if ((typeof(el) == "object") && el.xtype && el.xns) {
37689                 el = Roo.factory(el, Roo.menu);
37690             }
37691             
37692             if(el.render){ // some kind of Item
37693                 item = this.addItem(el);
37694             }else if(typeof el == "string"){ // string
37695                 if(el == "separator" || el == "-"){
37696                     item = this.addSeparator();
37697                 }else{
37698                     item = this.addText(el);
37699                 }
37700             }else if(el.tagName || el.el){ // element
37701                 item = this.addElement(el);
37702             }else if(typeof el == "object"){ // must be menu item config?
37703                 item = this.addMenuItem(el);
37704             }
37705         }
37706         return item;
37707     },
37708
37709     /**
37710      * Returns this menu's underlying {@link Roo.Element} object
37711      * @return {Roo.Element} The element
37712      */
37713     getEl : function(){
37714         if(!this.el){
37715             this.render();
37716         }
37717         return this.el;
37718     },
37719
37720     /**
37721      * Adds a separator bar to the menu
37722      * @return {Roo.menu.Item} The menu item that was added
37723      */
37724     addSeparator : function(){
37725         return this.addItem(new Roo.menu.Separator());
37726     },
37727
37728     /**
37729      * Adds an {@link Roo.Element} object to the menu
37730      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37731      * @return {Roo.menu.Item} The menu item that was added
37732      */
37733     addElement : function(el){
37734         return this.addItem(new Roo.menu.BaseItem(el));
37735     },
37736
37737     /**
37738      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37739      * @param {Roo.menu.Item} item The menu item to add
37740      * @return {Roo.menu.Item} The menu item that was added
37741      */
37742     addItem : function(item){
37743         this.items.add(item);
37744         if(this.ul){
37745             var li = document.createElement("li");
37746             li.className = "x-menu-list-item";
37747             this.ul.dom.appendChild(li);
37748             item.render(li, this);
37749             this.delayAutoWidth();
37750         }
37751         return item;
37752     },
37753
37754     /**
37755      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37756      * @param {Object} config A MenuItem config object
37757      * @return {Roo.menu.Item} The menu item that was added
37758      */
37759     addMenuItem : function(config){
37760         if(!(config instanceof Roo.menu.Item)){
37761             if(typeof config.checked == "boolean"){ // must be check menu item config?
37762                 config = new Roo.menu.CheckItem(config);
37763             }else{
37764                 config = new Roo.menu.Item(config);
37765             }
37766         }
37767         return this.addItem(config);
37768     },
37769
37770     /**
37771      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37772      * @param {String} text The text to display in the menu item
37773      * @return {Roo.menu.Item} The menu item that was added
37774      */
37775     addText : function(text){
37776         return this.addItem(new Roo.menu.TextItem({ text : text }));
37777     },
37778
37779     /**
37780      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37781      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37782      * @param {Roo.menu.Item} item The menu item to add
37783      * @return {Roo.menu.Item} The menu item that was added
37784      */
37785     insert : function(index, item){
37786         this.items.insert(index, item);
37787         if(this.ul){
37788             var li = document.createElement("li");
37789             li.className = "x-menu-list-item";
37790             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37791             item.render(li, this);
37792             this.delayAutoWidth();
37793         }
37794         return item;
37795     },
37796
37797     /**
37798      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37799      * @param {Roo.menu.Item} item The menu item to remove
37800      */
37801     remove : function(item){
37802         this.items.removeKey(item.id);
37803         item.destroy();
37804     },
37805
37806     /**
37807      * Removes and destroys all items in the menu
37808      */
37809     removeAll : function(){
37810         var f;
37811         while(f = this.items.first()){
37812             this.remove(f);
37813         }
37814     }
37815 });
37816
37817 // MenuNav is a private utility class used internally by the Menu
37818 Roo.menu.MenuNav = function(menu){
37819     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37820     this.scope = this.menu = menu;
37821 };
37822
37823 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37824     doRelay : function(e, h){
37825         var k = e.getKey();
37826         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37827             this.menu.tryActivate(0, 1);
37828             return false;
37829         }
37830         return h.call(this.scope || this, e, this.menu);
37831     },
37832
37833     up : function(e, m){
37834         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37835             m.tryActivate(m.items.length-1, -1);
37836         }
37837     },
37838
37839     down : function(e, m){
37840         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37841             m.tryActivate(0, 1);
37842         }
37843     },
37844
37845     right : function(e, m){
37846         if(m.activeItem){
37847             m.activeItem.expandMenu(true);
37848         }
37849     },
37850
37851     left : function(e, m){
37852         m.hide();
37853         if(m.parentMenu && m.parentMenu.activeItem){
37854             m.parentMenu.activeItem.activate();
37855         }
37856     },
37857
37858     enter : function(e, m){
37859         if(m.activeItem){
37860             e.stopPropagation();
37861             m.activeItem.onClick(e);
37862             m.fireEvent("click", this, m.activeItem);
37863             return true;
37864         }
37865     }
37866 });/*
37867  * Based on:
37868  * Ext JS Library 1.1.1
37869  * Copyright(c) 2006-2007, Ext JS, LLC.
37870  *
37871  * Originally Released Under LGPL - original licence link has changed is not relivant.
37872  *
37873  * Fork - LGPL
37874  * <script type="text/javascript">
37875  */
37876  
37877 /**
37878  * @class Roo.menu.MenuMgr
37879  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37880  * @singleton
37881  */
37882 Roo.menu.MenuMgr = function(){
37883    var menus, active, groups = {}, attached = false, lastShow = new Date();
37884
37885    // private - called when first menu is created
37886    function init(){
37887        menus = {};
37888        active = new Roo.util.MixedCollection();
37889        Roo.get(document).addKeyListener(27, function(){
37890            if(active.length > 0){
37891                hideAll();
37892            }
37893        });
37894    }
37895
37896    // private
37897    function hideAll(){
37898        if(active && active.length > 0){
37899            var c = active.clone();
37900            c.each(function(m){
37901                m.hide();
37902            });
37903        }
37904    }
37905
37906    // private
37907    function onHide(m){
37908        active.remove(m);
37909        if(active.length < 1){
37910            Roo.get(document).un("mousedown", onMouseDown);
37911            attached = false;
37912        }
37913    }
37914
37915    // private
37916    function onShow(m){
37917        var last = active.last();
37918        lastShow = new Date();
37919        active.add(m);
37920        if(!attached){
37921            Roo.get(document).on("mousedown", onMouseDown);
37922            attached = true;
37923        }
37924        if(m.parentMenu){
37925           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37926           m.parentMenu.activeChild = m;
37927        }else if(last && last.isVisible()){
37928           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37929        }
37930    }
37931
37932    // private
37933    function onBeforeHide(m){
37934        if(m.activeChild){
37935            m.activeChild.hide();
37936        }
37937        if(m.autoHideTimer){
37938            clearTimeout(m.autoHideTimer);
37939            delete m.autoHideTimer;
37940        }
37941    }
37942
37943    // private
37944    function onBeforeShow(m){
37945        var pm = m.parentMenu;
37946        if(!pm && !m.allowOtherMenus){
37947            hideAll();
37948        }else if(pm && pm.activeChild && active != m){
37949            pm.activeChild.hide();
37950        }
37951    }
37952
37953    // private
37954    function onMouseDown(e){
37955        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37956            hideAll();
37957        }
37958    }
37959
37960    // private
37961    function onBeforeCheck(mi, state){
37962        if(state){
37963            var g = groups[mi.group];
37964            for(var i = 0, l = g.length; i < l; i++){
37965                if(g[i] != mi){
37966                    g[i].setChecked(false);
37967                }
37968            }
37969        }
37970    }
37971
37972    return {
37973
37974        /**
37975         * Hides all menus that are currently visible
37976         */
37977        hideAll : function(){
37978             hideAll();  
37979        },
37980
37981        // private
37982        register : function(menu){
37983            if(!menus){
37984                init();
37985            }
37986            menus[menu.id] = menu;
37987            menu.on("beforehide", onBeforeHide);
37988            menu.on("hide", onHide);
37989            menu.on("beforeshow", onBeforeShow);
37990            menu.on("show", onShow);
37991            var g = menu.group;
37992            if(g && menu.events["checkchange"]){
37993                if(!groups[g]){
37994                    groups[g] = [];
37995                }
37996                groups[g].push(menu);
37997                menu.on("checkchange", onCheck);
37998            }
37999        },
38000
38001         /**
38002          * Returns a {@link Roo.menu.Menu} object
38003          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
38004          * be used to generate and return a new Menu instance.
38005          */
38006        get : function(menu){
38007            if(typeof menu == "string"){ // menu id
38008                return menus[menu];
38009            }else if(menu.events){  // menu instance
38010                return menu;
38011            }else if(typeof menu.length == 'number'){ // array of menu items?
38012                return new Roo.menu.Menu({items:menu});
38013            }else{ // otherwise, must be a config
38014                return new Roo.menu.Menu(menu);
38015            }
38016        },
38017
38018        // private
38019        unregister : function(menu){
38020            delete menus[menu.id];
38021            menu.un("beforehide", onBeforeHide);
38022            menu.un("hide", onHide);
38023            menu.un("beforeshow", onBeforeShow);
38024            menu.un("show", onShow);
38025            var g = menu.group;
38026            if(g && menu.events["checkchange"]){
38027                groups[g].remove(menu);
38028                menu.un("checkchange", onCheck);
38029            }
38030        },
38031
38032        // private
38033        registerCheckable : function(menuItem){
38034            var g = menuItem.group;
38035            if(g){
38036                if(!groups[g]){
38037                    groups[g] = [];
38038                }
38039                groups[g].push(menuItem);
38040                menuItem.on("beforecheckchange", onBeforeCheck);
38041            }
38042        },
38043
38044        // private
38045        unregisterCheckable : function(menuItem){
38046            var g = menuItem.group;
38047            if(g){
38048                groups[g].remove(menuItem);
38049                menuItem.un("beforecheckchange", onBeforeCheck);
38050            }
38051        }
38052    };
38053 }();/*
38054  * Based on:
38055  * Ext JS Library 1.1.1
38056  * Copyright(c) 2006-2007, Ext JS, LLC.
38057  *
38058  * Originally Released Under LGPL - original licence link has changed is not relivant.
38059  *
38060  * Fork - LGPL
38061  * <script type="text/javascript">
38062  */
38063  
38064
38065 /**
38066  * @class Roo.menu.BaseItem
38067  * @extends Roo.Component
38068  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38069  * management and base configuration options shared by all menu components.
38070  * @constructor
38071  * Creates a new BaseItem
38072  * @param {Object} config Configuration options
38073  */
38074 Roo.menu.BaseItem = function(config){
38075     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38076
38077     this.addEvents({
38078         /**
38079          * @event click
38080          * Fires when this item is clicked
38081          * @param {Roo.menu.BaseItem} this
38082          * @param {Roo.EventObject} e
38083          */
38084         click: true,
38085         /**
38086          * @event activate
38087          * Fires when this item is activated
38088          * @param {Roo.menu.BaseItem} this
38089          */
38090         activate : true,
38091         /**
38092          * @event deactivate
38093          * Fires when this item is deactivated
38094          * @param {Roo.menu.BaseItem} this
38095          */
38096         deactivate : true
38097     });
38098
38099     if(this.handler){
38100         this.on("click", this.handler, this.scope, true);
38101     }
38102 };
38103
38104 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38105     /**
38106      * @cfg {Function} handler
38107      * A function that will handle the click event of this menu item (defaults to undefined)
38108      */
38109     /**
38110      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38111      */
38112     canActivate : false,
38113     
38114      /**
38115      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38116      */
38117     hidden: false,
38118     
38119     /**
38120      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38121      */
38122     activeClass : "x-menu-item-active",
38123     /**
38124      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38125      */
38126     hideOnClick : true,
38127     /**
38128      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38129      */
38130     hideDelay : 100,
38131
38132     // private
38133     ctype: "Roo.menu.BaseItem",
38134
38135     // private
38136     actionMode : "container",
38137
38138     // private
38139     render : function(container, parentMenu){
38140         this.parentMenu = parentMenu;
38141         Roo.menu.BaseItem.superclass.render.call(this, container);
38142         this.container.menuItemId = this.id;
38143     },
38144
38145     // private
38146     onRender : function(container, position){
38147         this.el = Roo.get(this.el);
38148         container.dom.appendChild(this.el.dom);
38149     },
38150
38151     // private
38152     onClick : function(e){
38153         if(!this.disabled && this.fireEvent("click", this, e) !== false
38154                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38155             this.handleClick(e);
38156         }else{
38157             e.stopEvent();
38158         }
38159     },
38160
38161     // private
38162     activate : function(){
38163         if(this.disabled){
38164             return false;
38165         }
38166         var li = this.container;
38167         li.addClass(this.activeClass);
38168         this.region = li.getRegion().adjust(2, 2, -2, -2);
38169         this.fireEvent("activate", this);
38170         return true;
38171     },
38172
38173     // private
38174     deactivate : function(){
38175         this.container.removeClass(this.activeClass);
38176         this.fireEvent("deactivate", this);
38177     },
38178
38179     // private
38180     shouldDeactivate : function(e){
38181         return !this.region || !this.region.contains(e.getPoint());
38182     },
38183
38184     // private
38185     handleClick : function(e){
38186         if(this.hideOnClick){
38187             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38188         }
38189     },
38190
38191     // private
38192     expandMenu : function(autoActivate){
38193         // do nothing
38194     },
38195
38196     // private
38197     hideMenu : function(){
38198         // do nothing
38199     }
38200 });/*
38201  * Based on:
38202  * Ext JS Library 1.1.1
38203  * Copyright(c) 2006-2007, Ext JS, LLC.
38204  *
38205  * Originally Released Under LGPL - original licence link has changed is not relivant.
38206  *
38207  * Fork - LGPL
38208  * <script type="text/javascript">
38209  */
38210  
38211 /**
38212  * @class Roo.menu.Adapter
38213  * @extends Roo.menu.BaseItem
38214  * 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.
38215  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38216  * @constructor
38217  * Creates a new Adapter
38218  * @param {Object} config Configuration options
38219  */
38220 Roo.menu.Adapter = function(component, config){
38221     Roo.menu.Adapter.superclass.constructor.call(this, config);
38222     this.component = component;
38223 };
38224 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38225     // private
38226     canActivate : true,
38227
38228     // private
38229     onRender : function(container, position){
38230         this.component.render(container);
38231         this.el = this.component.getEl();
38232     },
38233
38234     // private
38235     activate : function(){
38236         if(this.disabled){
38237             return false;
38238         }
38239         this.component.focus();
38240         this.fireEvent("activate", this);
38241         return true;
38242     },
38243
38244     // private
38245     deactivate : function(){
38246         this.fireEvent("deactivate", this);
38247     },
38248
38249     // private
38250     disable : function(){
38251         this.component.disable();
38252         Roo.menu.Adapter.superclass.disable.call(this);
38253     },
38254
38255     // private
38256     enable : function(){
38257         this.component.enable();
38258         Roo.menu.Adapter.superclass.enable.call(this);
38259     }
38260 });/*
38261  * Based on:
38262  * Ext JS Library 1.1.1
38263  * Copyright(c) 2006-2007, Ext JS, LLC.
38264  *
38265  * Originally Released Under LGPL - original licence link has changed is not relivant.
38266  *
38267  * Fork - LGPL
38268  * <script type="text/javascript">
38269  */
38270
38271 /**
38272  * @class Roo.menu.TextItem
38273  * @extends Roo.menu.BaseItem
38274  * Adds a static text string to a menu, usually used as either a heading or group separator.
38275  * Note: old style constructor with text is still supported.
38276  * 
38277  * @constructor
38278  * Creates a new TextItem
38279  * @param {Object} cfg Configuration
38280  */
38281 Roo.menu.TextItem = function(cfg){
38282     if (typeof(cfg) == 'string') {
38283         this.text = cfg;
38284     } else {
38285         Roo.apply(this,cfg);
38286     }
38287     
38288     Roo.menu.TextItem.superclass.constructor.call(this);
38289 };
38290
38291 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38292     /**
38293      * @cfg {Boolean} text Text to show on item.
38294      */
38295     text : '',
38296     
38297     /**
38298      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38299      */
38300     hideOnClick : false,
38301     /**
38302      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38303      */
38304     itemCls : "x-menu-text",
38305
38306     // private
38307     onRender : function(){
38308         var s = document.createElement("span");
38309         s.className = this.itemCls;
38310         s.innerHTML = this.text;
38311         this.el = s;
38312         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38313     }
38314 });/*
38315  * Based on:
38316  * Ext JS Library 1.1.1
38317  * Copyright(c) 2006-2007, Ext JS, LLC.
38318  *
38319  * Originally Released Under LGPL - original licence link has changed is not relivant.
38320  *
38321  * Fork - LGPL
38322  * <script type="text/javascript">
38323  */
38324
38325 /**
38326  * @class Roo.menu.Separator
38327  * @extends Roo.menu.BaseItem
38328  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38329  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38330  * @constructor
38331  * @param {Object} config Configuration options
38332  */
38333 Roo.menu.Separator = function(config){
38334     Roo.menu.Separator.superclass.constructor.call(this, config);
38335 };
38336
38337 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38338     /**
38339      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38340      */
38341     itemCls : "x-menu-sep",
38342     /**
38343      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38344      */
38345     hideOnClick : false,
38346
38347     // private
38348     onRender : function(li){
38349         var s = document.createElement("span");
38350         s.className = this.itemCls;
38351         s.innerHTML = "&#160;";
38352         this.el = s;
38353         li.addClass("x-menu-sep-li");
38354         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38355     }
38356 });/*
38357  * Based on:
38358  * Ext JS Library 1.1.1
38359  * Copyright(c) 2006-2007, Ext JS, LLC.
38360  *
38361  * Originally Released Under LGPL - original licence link has changed is not relivant.
38362  *
38363  * Fork - LGPL
38364  * <script type="text/javascript">
38365  */
38366 /**
38367  * @class Roo.menu.Item
38368  * @extends Roo.menu.BaseItem
38369  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38370  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38371  * activation and click handling.
38372  * @constructor
38373  * Creates a new Item
38374  * @param {Object} config Configuration options
38375  */
38376 Roo.menu.Item = function(config){
38377     Roo.menu.Item.superclass.constructor.call(this, config);
38378     if(this.menu){
38379         this.menu = Roo.menu.MenuMgr.get(this.menu);
38380     }
38381 };
38382 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38383     
38384     /**
38385      * @cfg {String} text
38386      * The text to show on the menu item.
38387      */
38388     text: '',
38389      /**
38390      * @cfg {String} HTML to render in menu
38391      * The text to show on the menu item (HTML version).
38392      */
38393     html: '',
38394     /**
38395      * @cfg {String} icon
38396      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38397      */
38398     icon: undefined,
38399     /**
38400      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38401      */
38402     itemCls : "x-menu-item",
38403     /**
38404      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38405      */
38406     canActivate : true,
38407     /**
38408      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38409      */
38410     showDelay: 200,
38411     // doc'd in BaseItem
38412     hideDelay: 200,
38413
38414     // private
38415     ctype: "Roo.menu.Item",
38416     
38417     // private
38418     onRender : function(container, position){
38419         var el = document.createElement("a");
38420         el.hideFocus = true;
38421         el.unselectable = "on";
38422         el.href = this.href || "#";
38423         if(this.hrefTarget){
38424             el.target = this.hrefTarget;
38425         }
38426         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38427         
38428         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38429         
38430         el.innerHTML = String.format(
38431                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38432                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38433         this.el = el;
38434         Roo.menu.Item.superclass.onRender.call(this, container, position);
38435     },
38436
38437     /**
38438      * Sets the text to display in this menu item
38439      * @param {String} text The text to display
38440      * @param {Boolean} isHTML true to indicate text is pure html.
38441      */
38442     setText : function(text, isHTML){
38443         if (isHTML) {
38444             this.html = text;
38445         } else {
38446             this.text = text;
38447             this.html = '';
38448         }
38449         if(this.rendered){
38450             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38451      
38452             this.el.update(String.format(
38453                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38454                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38455             this.parentMenu.autoWidth();
38456         }
38457     },
38458
38459     // private
38460     handleClick : function(e){
38461         if(!this.href){ // if no link defined, stop the event automatically
38462             e.stopEvent();
38463         }
38464         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38465     },
38466
38467     // private
38468     activate : function(autoExpand){
38469         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38470             this.focus();
38471             if(autoExpand){
38472                 this.expandMenu();
38473             }
38474         }
38475         return true;
38476     },
38477
38478     // private
38479     shouldDeactivate : function(e){
38480         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38481             if(this.menu && this.menu.isVisible()){
38482                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38483             }
38484             return true;
38485         }
38486         return false;
38487     },
38488
38489     // private
38490     deactivate : function(){
38491         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38492         this.hideMenu();
38493     },
38494
38495     // private
38496     expandMenu : function(autoActivate){
38497         if(!this.disabled && this.menu){
38498             clearTimeout(this.hideTimer);
38499             delete this.hideTimer;
38500             if(!this.menu.isVisible() && !this.showTimer){
38501                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38502             }else if (this.menu.isVisible() && autoActivate){
38503                 this.menu.tryActivate(0, 1);
38504             }
38505         }
38506     },
38507
38508     // private
38509     deferExpand : function(autoActivate){
38510         delete this.showTimer;
38511         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38512         if(autoActivate){
38513             this.menu.tryActivate(0, 1);
38514         }
38515     },
38516
38517     // private
38518     hideMenu : function(){
38519         clearTimeout(this.showTimer);
38520         delete this.showTimer;
38521         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38522             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38523         }
38524     },
38525
38526     // private
38527     deferHide : function(){
38528         delete this.hideTimer;
38529         this.menu.hide();
38530     }
38531 });/*
38532  * Based on:
38533  * Ext JS Library 1.1.1
38534  * Copyright(c) 2006-2007, Ext JS, LLC.
38535  *
38536  * Originally Released Under LGPL - original licence link has changed is not relivant.
38537  *
38538  * Fork - LGPL
38539  * <script type="text/javascript">
38540  */
38541  
38542 /**
38543  * @class Roo.menu.CheckItem
38544  * @extends Roo.menu.Item
38545  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38546  * @constructor
38547  * Creates a new CheckItem
38548  * @param {Object} config Configuration options
38549  */
38550 Roo.menu.CheckItem = function(config){
38551     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38552     this.addEvents({
38553         /**
38554          * @event beforecheckchange
38555          * Fires before the checked value is set, providing an opportunity to cancel if needed
38556          * @param {Roo.menu.CheckItem} this
38557          * @param {Boolean} checked The new checked value that will be set
38558          */
38559         "beforecheckchange" : true,
38560         /**
38561          * @event checkchange
38562          * Fires after the checked value has been set
38563          * @param {Roo.menu.CheckItem} this
38564          * @param {Boolean} checked The checked value that was set
38565          */
38566         "checkchange" : true
38567     });
38568     if(this.checkHandler){
38569         this.on('checkchange', this.checkHandler, this.scope);
38570     }
38571 };
38572 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38573     /**
38574      * @cfg {String} group
38575      * All check items with the same group name will automatically be grouped into a single-select
38576      * radio button group (defaults to '')
38577      */
38578     /**
38579      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38580      */
38581     itemCls : "x-menu-item x-menu-check-item",
38582     /**
38583      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38584      */
38585     groupClass : "x-menu-group-item",
38586
38587     /**
38588      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38589      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38590      * initialized with checked = true will be rendered as checked.
38591      */
38592     checked: false,
38593
38594     // private
38595     ctype: "Roo.menu.CheckItem",
38596
38597     // private
38598     onRender : function(c){
38599         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38600         if(this.group){
38601             this.el.addClass(this.groupClass);
38602         }
38603         Roo.menu.MenuMgr.registerCheckable(this);
38604         if(this.checked){
38605             this.checked = false;
38606             this.setChecked(true, true);
38607         }
38608     },
38609
38610     // private
38611     destroy : function(){
38612         if(this.rendered){
38613             Roo.menu.MenuMgr.unregisterCheckable(this);
38614         }
38615         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38616     },
38617
38618     /**
38619      * Set the checked state of this item
38620      * @param {Boolean} checked The new checked value
38621      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38622      */
38623     setChecked : function(state, suppressEvent){
38624         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38625             if(this.container){
38626                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38627             }
38628             this.checked = state;
38629             if(suppressEvent !== true){
38630                 this.fireEvent("checkchange", this, state);
38631             }
38632         }
38633     },
38634
38635     // private
38636     handleClick : function(e){
38637        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38638            this.setChecked(!this.checked);
38639        }
38640        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38641     }
38642 });/*
38643  * Based on:
38644  * Ext JS Library 1.1.1
38645  * Copyright(c) 2006-2007, Ext JS, LLC.
38646  *
38647  * Originally Released Under LGPL - original licence link has changed is not relivant.
38648  *
38649  * Fork - LGPL
38650  * <script type="text/javascript">
38651  */
38652  
38653 /**
38654  * @class Roo.menu.DateItem
38655  * @extends Roo.menu.Adapter
38656  * A menu item that wraps the {@link Roo.DatPicker} component.
38657  * @constructor
38658  * Creates a new DateItem
38659  * @param {Object} config Configuration options
38660  */
38661 Roo.menu.DateItem = function(config){
38662     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38663     /** The Roo.DatePicker object @type Roo.DatePicker */
38664     this.picker = this.component;
38665     this.addEvents({select: true});
38666     
38667     this.picker.on("render", function(picker){
38668         picker.getEl().swallowEvent("click");
38669         picker.container.addClass("x-menu-date-item");
38670     });
38671
38672     this.picker.on("select", this.onSelect, this);
38673 };
38674
38675 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38676     // private
38677     onSelect : function(picker, date){
38678         this.fireEvent("select", this, date, picker);
38679         Roo.menu.DateItem.superclass.handleClick.call(this);
38680     }
38681 });/*
38682  * Based on:
38683  * Ext JS Library 1.1.1
38684  * Copyright(c) 2006-2007, Ext JS, LLC.
38685  *
38686  * Originally Released Under LGPL - original licence link has changed is not relivant.
38687  *
38688  * Fork - LGPL
38689  * <script type="text/javascript">
38690  */
38691  
38692 /**
38693  * @class Roo.menu.ColorItem
38694  * @extends Roo.menu.Adapter
38695  * A menu item that wraps the {@link Roo.ColorPalette} component.
38696  * @constructor
38697  * Creates a new ColorItem
38698  * @param {Object} config Configuration options
38699  */
38700 Roo.menu.ColorItem = function(config){
38701     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38702     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38703     this.palette = this.component;
38704     this.relayEvents(this.palette, ["select"]);
38705     if(this.selectHandler){
38706         this.on('select', this.selectHandler, this.scope);
38707     }
38708 };
38709 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38710  * Based on:
38711  * Ext JS Library 1.1.1
38712  * Copyright(c) 2006-2007, Ext JS, LLC.
38713  *
38714  * Originally Released Under LGPL - original licence link has changed is not relivant.
38715  *
38716  * Fork - LGPL
38717  * <script type="text/javascript">
38718  */
38719  
38720
38721 /**
38722  * @class Roo.menu.DateMenu
38723  * @extends Roo.menu.Menu
38724  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38725  * @constructor
38726  * Creates a new DateMenu
38727  * @param {Object} config Configuration options
38728  */
38729 Roo.menu.DateMenu = function(config){
38730     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38731     this.plain = true;
38732     var di = new Roo.menu.DateItem(config);
38733     this.add(di);
38734     /**
38735      * The {@link Roo.DatePicker} instance for this DateMenu
38736      * @type DatePicker
38737      */
38738     this.picker = di.picker;
38739     /**
38740      * @event select
38741      * @param {DatePicker} picker
38742      * @param {Date} date
38743      */
38744     this.relayEvents(di, ["select"]);
38745     this.on('beforeshow', function(){
38746         if(this.picker){
38747             this.picker.hideMonthPicker(false);
38748         }
38749     }, this);
38750 };
38751 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38752     cls:'x-date-menu'
38753 });/*
38754  * Based on:
38755  * Ext JS Library 1.1.1
38756  * Copyright(c) 2006-2007, Ext JS, LLC.
38757  *
38758  * Originally Released Under LGPL - original licence link has changed is not relivant.
38759  *
38760  * Fork - LGPL
38761  * <script type="text/javascript">
38762  */
38763  
38764
38765 /**
38766  * @class Roo.menu.ColorMenu
38767  * @extends Roo.menu.Menu
38768  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38769  * @constructor
38770  * Creates a new ColorMenu
38771  * @param {Object} config Configuration options
38772  */
38773 Roo.menu.ColorMenu = function(config){
38774     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38775     this.plain = true;
38776     var ci = new Roo.menu.ColorItem(config);
38777     this.add(ci);
38778     /**
38779      * The {@link Roo.ColorPalette} instance for this ColorMenu
38780      * @type ColorPalette
38781      */
38782     this.palette = ci.palette;
38783     /**
38784      * @event select
38785      * @param {ColorPalette} palette
38786      * @param {String} color
38787      */
38788     this.relayEvents(ci, ["select"]);
38789 };
38790 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38791  * Based on:
38792  * Ext JS Library 1.1.1
38793  * Copyright(c) 2006-2007, Ext JS, LLC.
38794  *
38795  * Originally Released Under LGPL - original licence link has changed is not relivant.
38796  *
38797  * Fork - LGPL
38798  * <script type="text/javascript">
38799  */
38800  
38801 /**
38802  * @class Roo.form.TextItem
38803  * @extends Roo.BoxComponent
38804  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38805  * @constructor
38806  * Creates a new TextItem
38807  * @param {Object} config Configuration options
38808  */
38809 Roo.form.TextItem = function(config){
38810     Roo.form.TextItem.superclass.constructor.call(this, config);
38811 };
38812
38813 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38814     
38815     /**
38816      * @cfg {String} tag the tag for this item (default div)
38817      */
38818     tag : 'div',
38819     /**
38820      * @cfg {String} html the content for this item
38821      */
38822     html : '',
38823     
38824     getAutoCreate : function()
38825     {
38826         var cfg = {
38827             id: this.id,
38828             tag: this.tag,
38829             html: this.html,
38830             cls: 'x-form-item'
38831         };
38832         
38833         return cfg;
38834         
38835     },
38836     
38837     onRender : function(ct, position)
38838     {
38839         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38840         
38841         if(!this.el){
38842             var cfg = this.getAutoCreate();
38843             if(!cfg.name){
38844                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38845             }
38846             if (!cfg.name.length) {
38847                 delete cfg.name;
38848             }
38849             this.el = ct.createChild(cfg, position);
38850         }
38851     },
38852     /*
38853      * setHTML
38854      * @param {String} html update the Contents of the element.
38855      */
38856     setHTML : function(html)
38857     {
38858         this.fieldEl.dom.innerHTML = html;
38859     }
38860     
38861 });/*
38862  * Based on:
38863  * Ext JS Library 1.1.1
38864  * Copyright(c) 2006-2007, Ext JS, LLC.
38865  *
38866  * Originally Released Under LGPL - original licence link has changed is not relivant.
38867  *
38868  * Fork - LGPL
38869  * <script type="text/javascript">
38870  */
38871  
38872 /**
38873  * @class Roo.form.Field
38874  * @extends Roo.BoxComponent
38875  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38876  * @constructor
38877  * Creates a new Field
38878  * @param {Object} config Configuration options
38879  */
38880 Roo.form.Field = function(config){
38881     Roo.form.Field.superclass.constructor.call(this, config);
38882 };
38883
38884 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38885     /**
38886      * @cfg {String} fieldLabel Label to use when rendering a form.
38887      */
38888        /**
38889      * @cfg {String} qtip Mouse over tip
38890      */
38891      
38892     /**
38893      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38894      */
38895     invalidClass : "x-form-invalid",
38896     /**
38897      * @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")
38898      */
38899     invalidText : "The value in this field is invalid",
38900     /**
38901      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38902      */
38903     focusClass : "x-form-focus",
38904     /**
38905      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38906       automatic validation (defaults to "keyup").
38907      */
38908     validationEvent : "keyup",
38909     /**
38910      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38911      */
38912     validateOnBlur : true,
38913     /**
38914      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38915      */
38916     validationDelay : 250,
38917     /**
38918      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38919      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38920      */
38921     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38922     /**
38923      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38924      */
38925     fieldClass : "x-form-field",
38926     /**
38927      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38928      *<pre>
38929 Value         Description
38930 -----------   ----------------------------------------------------------------------
38931 qtip          Display a quick tip when the user hovers over the field
38932 title         Display a default browser title attribute popup
38933 under         Add a block div beneath the field containing the error text
38934 side          Add an error icon to the right of the field with a popup on hover
38935 [element id]  Add the error text directly to the innerHTML of the specified element
38936 </pre>
38937      */
38938     msgTarget : 'qtip',
38939     /**
38940      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38941      */
38942     msgFx : 'normal',
38943
38944     /**
38945      * @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.
38946      */
38947     readOnly : false,
38948
38949     /**
38950      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38951      */
38952     disabled : false,
38953
38954     /**
38955      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38956      */
38957     inputType : undefined,
38958     
38959     /**
38960      * @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).
38961          */
38962         tabIndex : undefined,
38963         
38964     // private
38965     isFormField : true,
38966
38967     // private
38968     hasFocus : false,
38969     /**
38970      * @property {Roo.Element} fieldEl
38971      * Element Containing the rendered Field (with label etc.)
38972      */
38973     /**
38974      * @cfg {Mixed} value A value to initialize this field with.
38975      */
38976     value : undefined,
38977
38978     /**
38979      * @cfg {String} name The field's HTML name attribute.
38980      */
38981     /**
38982      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38983      */
38984     // private
38985     loadedValue : false,
38986      
38987      
38988         // private ??
38989         initComponent : function(){
38990         Roo.form.Field.superclass.initComponent.call(this);
38991         this.addEvents({
38992             /**
38993              * @event focus
38994              * Fires when this field receives input focus.
38995              * @param {Roo.form.Field} this
38996              */
38997             focus : true,
38998             /**
38999              * @event blur
39000              * Fires when this field loses input focus.
39001              * @param {Roo.form.Field} this
39002              */
39003             blur : true,
39004             /**
39005              * @event specialkey
39006              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
39007              * {@link Roo.EventObject#getKey} to determine which key was pressed.
39008              * @param {Roo.form.Field} this
39009              * @param {Roo.EventObject} e The event object
39010              */
39011             specialkey : true,
39012             /**
39013              * @event change
39014              * Fires just before the field blurs if the field value has changed.
39015              * @param {Roo.form.Field} this
39016              * @param {Mixed} newValue The new value
39017              * @param {Mixed} oldValue The original value
39018              */
39019             change : true,
39020             /**
39021              * @event invalid
39022              * Fires after the field has been marked as invalid.
39023              * @param {Roo.form.Field} this
39024              * @param {String} msg The validation message
39025              */
39026             invalid : true,
39027             /**
39028              * @event valid
39029              * Fires after the field has been validated with no errors.
39030              * @param {Roo.form.Field} this
39031              */
39032             valid : true,
39033              /**
39034              * @event keyup
39035              * Fires after the key up
39036              * @param {Roo.form.Field} this
39037              * @param {Roo.EventObject}  e The event Object
39038              */
39039             keyup : true
39040         });
39041     },
39042
39043     /**
39044      * Returns the name attribute of the field if available
39045      * @return {String} name The field name
39046      */
39047     getName: function(){
39048          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
39049     },
39050
39051     // private
39052     onRender : function(ct, position){
39053         Roo.form.Field.superclass.onRender.call(this, ct, position);
39054         if(!this.el){
39055             var cfg = this.getAutoCreate();
39056             if(!cfg.name){
39057                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39058             }
39059             if (!cfg.name.length) {
39060                 delete cfg.name;
39061             }
39062             if(this.inputType){
39063                 cfg.type = this.inputType;
39064             }
39065             this.el = ct.createChild(cfg, position);
39066         }
39067         var type = this.el.dom.type;
39068         if(type){
39069             if(type == 'password'){
39070                 type = 'text';
39071             }
39072             this.el.addClass('x-form-'+type);
39073         }
39074         if(this.readOnly){
39075             this.el.dom.readOnly = true;
39076         }
39077         if(this.tabIndex !== undefined){
39078             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39079         }
39080
39081         this.el.addClass([this.fieldClass, this.cls]);
39082         this.initValue();
39083     },
39084
39085     /**
39086      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39087      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39088      * @return {Roo.form.Field} this
39089      */
39090     applyTo : function(target){
39091         this.allowDomMove = false;
39092         this.el = Roo.get(target);
39093         this.render(this.el.dom.parentNode);
39094         return this;
39095     },
39096
39097     // private
39098     initValue : function(){
39099         if(this.value !== undefined){
39100             this.setValue(this.value);
39101         }else if(this.el.dom.value.length > 0){
39102             this.setValue(this.el.dom.value);
39103         }
39104     },
39105
39106     /**
39107      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39108      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39109      */
39110     isDirty : function() {
39111         if(this.disabled) {
39112             return false;
39113         }
39114         return String(this.getValue()) !== String(this.originalValue);
39115     },
39116
39117     /**
39118      * stores the current value in loadedValue
39119      */
39120     resetHasChanged : function()
39121     {
39122         this.loadedValue = String(this.getValue());
39123     },
39124     /**
39125      * checks the current value against the 'loaded' value.
39126      * Note - will return false if 'resetHasChanged' has not been called first.
39127      */
39128     hasChanged : function()
39129     {
39130         if(this.disabled || this.readOnly) {
39131             return false;
39132         }
39133         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39134     },
39135     
39136     
39137     
39138     // private
39139     afterRender : function(){
39140         Roo.form.Field.superclass.afterRender.call(this);
39141         this.initEvents();
39142     },
39143
39144     // private
39145     fireKey : function(e){
39146         //Roo.log('field ' + e.getKey());
39147         if(e.isNavKeyPress()){
39148             this.fireEvent("specialkey", this, e);
39149         }
39150     },
39151
39152     /**
39153      * Resets the current field value to the originally loaded value and clears any validation messages
39154      */
39155     reset : function(){
39156         this.setValue(this.resetValue);
39157         this.originalValue = this.getValue();
39158         this.clearInvalid();
39159     },
39160
39161     // private
39162     initEvents : function(){
39163         // safari killled keypress - so keydown is now used..
39164         this.el.on("keydown" , this.fireKey,  this);
39165         this.el.on("focus", this.onFocus,  this);
39166         this.el.on("blur", this.onBlur,  this);
39167         this.el.relayEvent('keyup', this);
39168
39169         // reference to original value for reset
39170         this.originalValue = this.getValue();
39171         this.resetValue =  this.getValue();
39172     },
39173
39174     // private
39175     onFocus : function(){
39176         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39177             this.el.addClass(this.focusClass);
39178         }
39179         if(!this.hasFocus){
39180             this.hasFocus = true;
39181             this.startValue = this.getValue();
39182             this.fireEvent("focus", this);
39183         }
39184     },
39185
39186     beforeBlur : Roo.emptyFn,
39187
39188     // private
39189     onBlur : function(){
39190         this.beforeBlur();
39191         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39192             this.el.removeClass(this.focusClass);
39193         }
39194         this.hasFocus = false;
39195         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39196             this.validate();
39197         }
39198         var v = this.getValue();
39199         if(String(v) !== String(this.startValue)){
39200             this.fireEvent('change', this, v, this.startValue);
39201         }
39202         this.fireEvent("blur", this);
39203     },
39204
39205     /**
39206      * Returns whether or not the field value is currently valid
39207      * @param {Boolean} preventMark True to disable marking the field invalid
39208      * @return {Boolean} True if the value is valid, else false
39209      */
39210     isValid : function(preventMark){
39211         if(this.disabled){
39212             return true;
39213         }
39214         var restore = this.preventMark;
39215         this.preventMark = preventMark === true;
39216         var v = this.validateValue(this.processValue(this.getRawValue()));
39217         this.preventMark = restore;
39218         return v;
39219     },
39220
39221     /**
39222      * Validates the field value
39223      * @return {Boolean} True if the value is valid, else false
39224      */
39225     validate : function(){
39226         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39227             this.clearInvalid();
39228             return true;
39229         }
39230         return false;
39231     },
39232
39233     processValue : function(value){
39234         return value;
39235     },
39236
39237     // private
39238     // Subclasses should provide the validation implementation by overriding this
39239     validateValue : function(value){
39240         return true;
39241     },
39242
39243     /**
39244      * Mark this field as invalid
39245      * @param {String} msg The validation message
39246      */
39247     markInvalid : function(msg){
39248         if(!this.rendered || this.preventMark){ // not rendered
39249             return;
39250         }
39251         
39252         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39253         
39254         obj.el.addClass(this.invalidClass);
39255         msg = msg || this.invalidText;
39256         switch(this.msgTarget){
39257             case 'qtip':
39258                 obj.el.dom.qtip = msg;
39259                 obj.el.dom.qclass = 'x-form-invalid-tip';
39260                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39261                     Roo.QuickTips.enable();
39262                 }
39263                 break;
39264             case 'title':
39265                 this.el.dom.title = msg;
39266                 break;
39267             case 'under':
39268                 if(!this.errorEl){
39269                     var elp = this.el.findParent('.x-form-element', 5, true);
39270                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39271                     this.errorEl.setWidth(elp.getWidth(true)-20);
39272                 }
39273                 this.errorEl.update(msg);
39274                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39275                 break;
39276             case 'side':
39277                 if(!this.errorIcon){
39278                     var elp = this.el.findParent('.x-form-element', 5, true);
39279                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39280                 }
39281                 this.alignErrorIcon();
39282                 this.errorIcon.dom.qtip = msg;
39283                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39284                 this.errorIcon.show();
39285                 this.on('resize', this.alignErrorIcon, this);
39286                 break;
39287             default:
39288                 var t = Roo.getDom(this.msgTarget);
39289                 t.innerHTML = msg;
39290                 t.style.display = this.msgDisplay;
39291                 break;
39292         }
39293         this.fireEvent('invalid', this, msg);
39294     },
39295
39296     // private
39297     alignErrorIcon : function(){
39298         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39299     },
39300
39301     /**
39302      * Clear any invalid styles/messages for this field
39303      */
39304     clearInvalid : function(){
39305         if(!this.rendered || this.preventMark){ // not rendered
39306             return;
39307         }
39308         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39309         
39310         obj.el.removeClass(this.invalidClass);
39311         switch(this.msgTarget){
39312             case 'qtip':
39313                 obj.el.dom.qtip = '';
39314                 break;
39315             case 'title':
39316                 this.el.dom.title = '';
39317                 break;
39318             case 'under':
39319                 if(this.errorEl){
39320                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39321                 }
39322                 break;
39323             case 'side':
39324                 if(this.errorIcon){
39325                     this.errorIcon.dom.qtip = '';
39326                     this.errorIcon.hide();
39327                     this.un('resize', this.alignErrorIcon, this);
39328                 }
39329                 break;
39330             default:
39331                 var t = Roo.getDom(this.msgTarget);
39332                 t.innerHTML = '';
39333                 t.style.display = 'none';
39334                 break;
39335         }
39336         this.fireEvent('valid', this);
39337     },
39338
39339     /**
39340      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39341      * @return {Mixed} value The field value
39342      */
39343     getRawValue : function(){
39344         var v = this.el.getValue();
39345         
39346         return v;
39347     },
39348
39349     /**
39350      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39351      * @return {Mixed} value The field value
39352      */
39353     getValue : function(){
39354         var v = this.el.getValue();
39355          
39356         return v;
39357     },
39358
39359     /**
39360      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39361      * @param {Mixed} value The value to set
39362      */
39363     setRawValue : function(v){
39364         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39365     },
39366
39367     /**
39368      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39369      * @param {Mixed} value The value to set
39370      */
39371     setValue : function(v){
39372         this.value = v;
39373         if(this.rendered){
39374             this.el.dom.value = (v === null || v === undefined ? '' : v);
39375              this.validate();
39376         }
39377     },
39378
39379     adjustSize : function(w, h){
39380         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39381         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39382         return s;
39383     },
39384
39385     adjustWidth : function(tag, w){
39386         tag = tag.toLowerCase();
39387         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39388             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39389                 if(tag == 'input'){
39390                     return w + 2;
39391                 }
39392                 if(tag == 'textarea'){
39393                     return w-2;
39394                 }
39395             }else if(Roo.isOpera){
39396                 if(tag == 'input'){
39397                     return w + 2;
39398                 }
39399                 if(tag == 'textarea'){
39400                     return w-2;
39401                 }
39402             }
39403         }
39404         return w;
39405     }
39406 });
39407
39408
39409 // anything other than normal should be considered experimental
39410 Roo.form.Field.msgFx = {
39411     normal : {
39412         show: function(msgEl, f){
39413             msgEl.setDisplayed('block');
39414         },
39415
39416         hide : function(msgEl, f){
39417             msgEl.setDisplayed(false).update('');
39418         }
39419     },
39420
39421     slide : {
39422         show: function(msgEl, f){
39423             msgEl.slideIn('t', {stopFx:true});
39424         },
39425
39426         hide : function(msgEl, f){
39427             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39428         }
39429     },
39430
39431     slideRight : {
39432         show: function(msgEl, f){
39433             msgEl.fixDisplay();
39434             msgEl.alignTo(f.el, 'tl-tr');
39435             msgEl.slideIn('l', {stopFx:true});
39436         },
39437
39438         hide : function(msgEl, f){
39439             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39440         }
39441     }
39442 };/*
39443  * Based on:
39444  * Ext JS Library 1.1.1
39445  * Copyright(c) 2006-2007, Ext JS, LLC.
39446  *
39447  * Originally Released Under LGPL - original licence link has changed is not relivant.
39448  *
39449  * Fork - LGPL
39450  * <script type="text/javascript">
39451  */
39452  
39453
39454 /**
39455  * @class Roo.form.TextField
39456  * @extends Roo.form.Field
39457  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39458  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39459  * @constructor
39460  * Creates a new TextField
39461  * @param {Object} config Configuration options
39462  */
39463 Roo.form.TextField = function(config){
39464     Roo.form.TextField.superclass.constructor.call(this, config);
39465     this.addEvents({
39466         /**
39467          * @event autosize
39468          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39469          * according to the default logic, but this event provides a hook for the developer to apply additional
39470          * logic at runtime to resize the field if needed.
39471              * @param {Roo.form.Field} this This text field
39472              * @param {Number} width The new field width
39473              */
39474         autosize : true
39475     });
39476 };
39477
39478 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39479     /**
39480      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39481      */
39482     grow : false,
39483     /**
39484      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39485      */
39486     growMin : 30,
39487     /**
39488      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39489      */
39490     growMax : 800,
39491     /**
39492      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39493      */
39494     vtype : null,
39495     /**
39496      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39497      */
39498     maskRe : null,
39499     /**
39500      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39501      */
39502     disableKeyFilter : false,
39503     /**
39504      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39505      */
39506     allowBlank : true,
39507     /**
39508      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39509      */
39510     minLength : 0,
39511     /**
39512      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39513      */
39514     maxLength : Number.MAX_VALUE,
39515     /**
39516      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39517      */
39518     minLengthText : "The minimum length for this field is {0}",
39519     /**
39520      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39521      */
39522     maxLengthText : "The maximum length for this field is {0}",
39523     /**
39524      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39525      */
39526     selectOnFocus : false,
39527     /**
39528      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39529      */    
39530     allowLeadingSpace : false,
39531     /**
39532      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39533      */
39534     blankText : "This field is required",
39535     /**
39536      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39537      * If available, this function will be called only after the basic validators all return true, and will be passed the
39538      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39539      */
39540     validator : null,
39541     /**
39542      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39543      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39544      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39545      */
39546     regex : null,
39547     /**
39548      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39549      */
39550     regexText : "",
39551     /**
39552      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39553      */
39554     emptyText : null,
39555    
39556
39557     // private
39558     initEvents : function()
39559     {
39560         if (this.emptyText) {
39561             this.el.attr('placeholder', this.emptyText);
39562         }
39563         
39564         Roo.form.TextField.superclass.initEvents.call(this);
39565         if(this.validationEvent == 'keyup'){
39566             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39567             this.el.on('keyup', this.filterValidation, this);
39568         }
39569         else if(this.validationEvent !== false){
39570             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39571         }
39572         
39573         if(this.selectOnFocus){
39574             this.on("focus", this.preFocus, this);
39575         }
39576         if (!this.allowLeadingSpace) {
39577             this.on('blur', this.cleanLeadingSpace, this);
39578         }
39579         
39580         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39581             this.el.on("keypress", this.filterKeys, this);
39582         }
39583         if(this.grow){
39584             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39585             this.el.on("click", this.autoSize,  this);
39586         }
39587         if(this.el.is('input[type=password]') && Roo.isSafari){
39588             this.el.on('keydown', this.SafariOnKeyDown, this);
39589         }
39590     },
39591
39592     processValue : function(value){
39593         if(this.stripCharsRe){
39594             var newValue = value.replace(this.stripCharsRe, '');
39595             if(newValue !== value){
39596                 this.setRawValue(newValue);
39597                 return newValue;
39598             }
39599         }
39600         return value;
39601     },
39602
39603     filterValidation : function(e){
39604         if(!e.isNavKeyPress()){
39605             this.validationTask.delay(this.validationDelay);
39606         }
39607     },
39608
39609     // private
39610     onKeyUp : function(e){
39611         if(!e.isNavKeyPress()){
39612             this.autoSize();
39613         }
39614     },
39615     // private - clean the leading white space
39616     cleanLeadingSpace : function(e)
39617     {
39618         if ( this.inputType == 'file') {
39619             return;
39620         }
39621         
39622         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39623     },
39624     /**
39625      * Resets the current field value to the originally-loaded value and clears any validation messages.
39626      *  
39627      */
39628     reset : function(){
39629         Roo.form.TextField.superclass.reset.call(this);
39630        
39631     }, 
39632     // private
39633     preFocus : function(){
39634         
39635         if(this.selectOnFocus){
39636             this.el.dom.select();
39637         }
39638     },
39639
39640     
39641     // private
39642     filterKeys : function(e){
39643         var k = e.getKey();
39644         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39645             return;
39646         }
39647         var c = e.getCharCode(), cc = String.fromCharCode(c);
39648         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39649             return;
39650         }
39651         if(!this.maskRe.test(cc)){
39652             e.stopEvent();
39653         }
39654     },
39655
39656     setValue : function(v){
39657         
39658         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39659         
39660         this.autoSize();
39661     },
39662
39663     /**
39664      * Validates a value according to the field's validation rules and marks the field as invalid
39665      * if the validation fails
39666      * @param {Mixed} value The value to validate
39667      * @return {Boolean} True if the value is valid, else false
39668      */
39669     validateValue : function(value){
39670         if(value.length < 1)  { // if it's blank
39671              if(this.allowBlank){
39672                 this.clearInvalid();
39673                 return true;
39674              }else{
39675                 this.markInvalid(this.blankText);
39676                 return false;
39677              }
39678         }
39679         if(value.length < this.minLength){
39680             this.markInvalid(String.format(this.minLengthText, this.minLength));
39681             return false;
39682         }
39683         if(value.length > this.maxLength){
39684             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39685             return false;
39686         }
39687         if(this.vtype){
39688             var vt = Roo.form.VTypes;
39689             if(!vt[this.vtype](value, this)){
39690                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39691                 return false;
39692             }
39693         }
39694         if(typeof this.validator == "function"){
39695             var msg = this.validator(value);
39696             if(msg !== true){
39697                 this.markInvalid(msg);
39698                 return false;
39699             }
39700         }
39701         if(this.regex && !this.regex.test(value)){
39702             this.markInvalid(this.regexText);
39703             return false;
39704         }
39705         return true;
39706     },
39707
39708     /**
39709      * Selects text in this field
39710      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39711      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39712      */
39713     selectText : function(start, end){
39714         var v = this.getRawValue();
39715         if(v.length > 0){
39716             start = start === undefined ? 0 : start;
39717             end = end === undefined ? v.length : end;
39718             var d = this.el.dom;
39719             if(d.setSelectionRange){
39720                 d.setSelectionRange(start, end);
39721             }else if(d.createTextRange){
39722                 var range = d.createTextRange();
39723                 range.moveStart("character", start);
39724                 range.moveEnd("character", v.length-end);
39725                 range.select();
39726             }
39727         }
39728     },
39729
39730     /**
39731      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39732      * This only takes effect if grow = true, and fires the autosize event.
39733      */
39734     autoSize : function(){
39735         if(!this.grow || !this.rendered){
39736             return;
39737         }
39738         if(!this.metrics){
39739             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39740         }
39741         var el = this.el;
39742         var v = el.dom.value;
39743         var d = document.createElement('div');
39744         d.appendChild(document.createTextNode(v));
39745         v = d.innerHTML;
39746         d = null;
39747         v += "&#160;";
39748         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39749         this.el.setWidth(w);
39750         this.fireEvent("autosize", this, w);
39751     },
39752     
39753     // private
39754     SafariOnKeyDown : function(event)
39755     {
39756         // this is a workaround for a password hang bug on chrome/ webkit.
39757         
39758         var isSelectAll = false;
39759         
39760         if(this.el.dom.selectionEnd > 0){
39761             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39762         }
39763         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39764             event.preventDefault();
39765             this.setValue('');
39766             return;
39767         }
39768         
39769         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39770             
39771             event.preventDefault();
39772             // this is very hacky as keydown always get's upper case.
39773             
39774             var cc = String.fromCharCode(event.getCharCode());
39775             
39776             
39777             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39778             
39779         }
39780         
39781         
39782     }
39783 });/*
39784  * Based on:
39785  * Ext JS Library 1.1.1
39786  * Copyright(c) 2006-2007, Ext JS, LLC.
39787  *
39788  * Originally Released Under LGPL - original licence link has changed is not relivant.
39789  *
39790  * Fork - LGPL
39791  * <script type="text/javascript">
39792  */
39793  
39794 /**
39795  * @class Roo.form.Hidden
39796  * @extends Roo.form.TextField
39797  * Simple Hidden element used on forms 
39798  * 
39799  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39800  * 
39801  * @constructor
39802  * Creates a new Hidden form element.
39803  * @param {Object} config Configuration options
39804  */
39805
39806
39807
39808 // easy hidden field...
39809 Roo.form.Hidden = function(config){
39810     Roo.form.Hidden.superclass.constructor.call(this, config);
39811 };
39812   
39813 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39814     fieldLabel:      '',
39815     inputType:      'hidden',
39816     width:          50,
39817     allowBlank:     true,
39818     labelSeparator: '',
39819     hidden:         true,
39820     itemCls :       'x-form-item-display-none'
39821
39822
39823 });
39824
39825
39826 /*
39827  * Based on:
39828  * Ext JS Library 1.1.1
39829  * Copyright(c) 2006-2007, Ext JS, LLC.
39830  *
39831  * Originally Released Under LGPL - original licence link has changed is not relivant.
39832  *
39833  * Fork - LGPL
39834  * <script type="text/javascript">
39835  */
39836  
39837 /**
39838  * @class Roo.form.TriggerField
39839  * @extends Roo.form.TextField
39840  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39841  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39842  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39843  * for which you can provide a custom implementation.  For example:
39844  * <pre><code>
39845 var trigger = new Roo.form.TriggerField();
39846 trigger.onTriggerClick = myTriggerFn;
39847 trigger.applyTo('my-field');
39848 </code></pre>
39849  *
39850  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39851  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39852  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39853  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39854  * @constructor
39855  * Create a new TriggerField.
39856  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39857  * to the base TextField)
39858  */
39859 Roo.form.TriggerField = function(config){
39860     this.mimicing = false;
39861     Roo.form.TriggerField.superclass.constructor.call(this, config);
39862 };
39863
39864 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39865     /**
39866      * @cfg {String} triggerClass A CSS class to apply to the trigger
39867      */
39868     /**
39869      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39870      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39871      */
39872     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39873     /**
39874      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39875      */
39876     hideTrigger:false,
39877
39878     /** @cfg {Boolean} grow @hide */
39879     /** @cfg {Number} growMin @hide */
39880     /** @cfg {Number} growMax @hide */
39881
39882     /**
39883      * @hide 
39884      * @method
39885      */
39886     autoSize: Roo.emptyFn,
39887     // private
39888     monitorTab : true,
39889     // private
39890     deferHeight : true,
39891
39892     
39893     actionMode : 'wrap',
39894     // private
39895     onResize : function(w, h){
39896         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39897         if(typeof w == 'number'){
39898             var x = w - this.trigger.getWidth();
39899             this.el.setWidth(this.adjustWidth('input', x));
39900             this.trigger.setStyle('left', x+'px');
39901         }
39902     },
39903
39904     // private
39905     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39906
39907     // private
39908     getResizeEl : function(){
39909         return this.wrap;
39910     },
39911
39912     // private
39913     getPositionEl : function(){
39914         return this.wrap;
39915     },
39916
39917     // private
39918     alignErrorIcon : function(){
39919         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39920     },
39921
39922     // private
39923     onRender : function(ct, position){
39924         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39925         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39926         this.trigger = this.wrap.createChild(this.triggerConfig ||
39927                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39928         if(this.hideTrigger){
39929             this.trigger.setDisplayed(false);
39930         }
39931         this.initTrigger();
39932         if(!this.width){
39933             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39934         }
39935     },
39936
39937     // private
39938     initTrigger : function(){
39939         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39940         this.trigger.addClassOnOver('x-form-trigger-over');
39941         this.trigger.addClassOnClick('x-form-trigger-click');
39942     },
39943
39944     // private
39945     onDestroy : function(){
39946         if(this.trigger){
39947             this.trigger.removeAllListeners();
39948             this.trigger.remove();
39949         }
39950         if(this.wrap){
39951             this.wrap.remove();
39952         }
39953         Roo.form.TriggerField.superclass.onDestroy.call(this);
39954     },
39955
39956     // private
39957     onFocus : function(){
39958         Roo.form.TriggerField.superclass.onFocus.call(this);
39959         if(!this.mimicing){
39960             this.wrap.addClass('x-trigger-wrap-focus');
39961             this.mimicing = true;
39962             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39963             if(this.monitorTab){
39964                 this.el.on("keydown", this.checkTab, this);
39965             }
39966         }
39967     },
39968
39969     // private
39970     checkTab : function(e){
39971         if(e.getKey() == e.TAB){
39972             this.triggerBlur();
39973         }
39974     },
39975
39976     // private
39977     onBlur : function(){
39978         // do nothing
39979     },
39980
39981     // private
39982     mimicBlur : function(e, t){
39983         if(!this.wrap.contains(t) && this.validateBlur()){
39984             this.triggerBlur();
39985         }
39986     },
39987
39988     // private
39989     triggerBlur : function(){
39990         this.mimicing = false;
39991         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39992         if(this.monitorTab){
39993             this.el.un("keydown", this.checkTab, this);
39994         }
39995         this.wrap.removeClass('x-trigger-wrap-focus');
39996         Roo.form.TriggerField.superclass.onBlur.call(this);
39997     },
39998
39999     // private
40000     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
40001     validateBlur : function(e, t){
40002         return true;
40003     },
40004
40005     // private
40006     onDisable : function(){
40007         Roo.form.TriggerField.superclass.onDisable.call(this);
40008         if(this.wrap){
40009             this.wrap.addClass('x-item-disabled');
40010         }
40011     },
40012
40013     // private
40014     onEnable : function(){
40015         Roo.form.TriggerField.superclass.onEnable.call(this);
40016         if(this.wrap){
40017             this.wrap.removeClass('x-item-disabled');
40018         }
40019     },
40020
40021     // private
40022     onShow : function(){
40023         var ae = this.getActionEl();
40024         
40025         if(ae){
40026             ae.dom.style.display = '';
40027             ae.dom.style.visibility = 'visible';
40028         }
40029     },
40030
40031     // private
40032     
40033     onHide : function(){
40034         var ae = this.getActionEl();
40035         ae.dom.style.display = 'none';
40036     },
40037
40038     /**
40039      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
40040      * by an implementing function.
40041      * @method
40042      * @param {EventObject} e
40043      */
40044     onTriggerClick : Roo.emptyFn
40045 });
40046
40047 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
40048 // to be extended by an implementing class.  For an example of implementing this class, see the custom
40049 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
40050 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
40051     initComponent : function(){
40052         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40053
40054         this.triggerConfig = {
40055             tag:'span', cls:'x-form-twin-triggers', cn:[
40056             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40057             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40058         ]};
40059     },
40060
40061     getTrigger : function(index){
40062         return this.triggers[index];
40063     },
40064
40065     initTrigger : function(){
40066         var ts = this.trigger.select('.x-form-trigger', true);
40067         this.wrap.setStyle('overflow', 'hidden');
40068         var triggerField = this;
40069         ts.each(function(t, all, index){
40070             t.hide = function(){
40071                 var w = triggerField.wrap.getWidth();
40072                 this.dom.style.display = 'none';
40073                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40074             };
40075             t.show = function(){
40076                 var w = triggerField.wrap.getWidth();
40077                 this.dom.style.display = '';
40078                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40079             };
40080             var triggerIndex = 'Trigger'+(index+1);
40081
40082             if(this['hide'+triggerIndex]){
40083                 t.dom.style.display = 'none';
40084             }
40085             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40086             t.addClassOnOver('x-form-trigger-over');
40087             t.addClassOnClick('x-form-trigger-click');
40088         }, this);
40089         this.triggers = ts.elements;
40090     },
40091
40092     onTrigger1Click : Roo.emptyFn,
40093     onTrigger2Click : Roo.emptyFn
40094 });/*
40095  * Based on:
40096  * Ext JS Library 1.1.1
40097  * Copyright(c) 2006-2007, Ext JS, LLC.
40098  *
40099  * Originally Released Under LGPL - original licence link has changed is not relivant.
40100  *
40101  * Fork - LGPL
40102  * <script type="text/javascript">
40103  */
40104  
40105 /**
40106  * @class Roo.form.TextArea
40107  * @extends Roo.form.TextField
40108  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40109  * support for auto-sizing.
40110  * @constructor
40111  * Creates a new TextArea
40112  * @param {Object} config Configuration options
40113  */
40114 Roo.form.TextArea = function(config){
40115     Roo.form.TextArea.superclass.constructor.call(this, config);
40116     // these are provided exchanges for backwards compat
40117     // minHeight/maxHeight were replaced by growMin/growMax to be
40118     // compatible with TextField growing config values
40119     if(this.minHeight !== undefined){
40120         this.growMin = this.minHeight;
40121     }
40122     if(this.maxHeight !== undefined){
40123         this.growMax = this.maxHeight;
40124     }
40125 };
40126
40127 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40128     /**
40129      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40130      */
40131     growMin : 60,
40132     /**
40133      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40134      */
40135     growMax: 1000,
40136     /**
40137      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40138      * in the field (equivalent to setting overflow: hidden, defaults to false)
40139      */
40140     preventScrollbars: false,
40141     /**
40142      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40143      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40144      */
40145
40146     // private
40147     onRender : function(ct, position){
40148         if(!this.el){
40149             this.defaultAutoCreate = {
40150                 tag: "textarea",
40151                 style:"width:300px;height:60px;",
40152                 autocomplete: "new-password"
40153             };
40154         }
40155         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40156         if(this.grow){
40157             this.textSizeEl = Roo.DomHelper.append(document.body, {
40158                 tag: "pre", cls: "x-form-grow-sizer"
40159             });
40160             if(this.preventScrollbars){
40161                 this.el.setStyle("overflow", "hidden");
40162             }
40163             this.el.setHeight(this.growMin);
40164         }
40165     },
40166
40167     onDestroy : function(){
40168         if(this.textSizeEl){
40169             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40170         }
40171         Roo.form.TextArea.superclass.onDestroy.call(this);
40172     },
40173
40174     // private
40175     onKeyUp : function(e){
40176         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40177             this.autoSize();
40178         }
40179     },
40180
40181     /**
40182      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40183      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40184      */
40185     autoSize : function(){
40186         if(!this.grow || !this.textSizeEl){
40187             return;
40188         }
40189         var el = this.el;
40190         var v = el.dom.value;
40191         var ts = this.textSizeEl;
40192
40193         ts.innerHTML = '';
40194         ts.appendChild(document.createTextNode(v));
40195         v = ts.innerHTML;
40196
40197         Roo.fly(ts).setWidth(this.el.getWidth());
40198         if(v.length < 1){
40199             v = "&#160;&#160;";
40200         }else{
40201             if(Roo.isIE){
40202                 v = v.replace(/\n/g, '<p>&#160;</p>');
40203             }
40204             v += "&#160;\n&#160;";
40205         }
40206         ts.innerHTML = v;
40207         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40208         if(h != this.lastHeight){
40209             this.lastHeight = h;
40210             this.el.setHeight(h);
40211             this.fireEvent("autosize", this, h);
40212         }
40213     }
40214 });/*
40215  * Based on:
40216  * Ext JS Library 1.1.1
40217  * Copyright(c) 2006-2007, Ext JS, LLC.
40218  *
40219  * Originally Released Under LGPL - original licence link has changed is not relivant.
40220  *
40221  * Fork - LGPL
40222  * <script type="text/javascript">
40223  */
40224  
40225
40226 /**
40227  * @class Roo.form.NumberField
40228  * @extends Roo.form.TextField
40229  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40230  * @constructor
40231  * Creates a new NumberField
40232  * @param {Object} config Configuration options
40233  */
40234 Roo.form.NumberField = function(config){
40235     Roo.form.NumberField.superclass.constructor.call(this, config);
40236 };
40237
40238 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40239     /**
40240      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40241      */
40242     fieldClass: "x-form-field x-form-num-field",
40243     /**
40244      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40245      */
40246     allowDecimals : true,
40247     /**
40248      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40249      */
40250     decimalSeparator : ".",
40251     /**
40252      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40253      */
40254     decimalPrecision : 2,
40255     /**
40256      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40257      */
40258     allowNegative : true,
40259     /**
40260      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40261      */
40262     minValue : Number.NEGATIVE_INFINITY,
40263     /**
40264      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40265      */
40266     maxValue : Number.MAX_VALUE,
40267     /**
40268      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40269      */
40270     minText : "The minimum value for this field is {0}",
40271     /**
40272      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40273      */
40274     maxText : "The maximum value for this field is {0}",
40275     /**
40276      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40277      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40278      */
40279     nanText : "{0} is not a valid number",
40280
40281     // private
40282     initEvents : function(){
40283         Roo.form.NumberField.superclass.initEvents.call(this);
40284         var allowed = "0123456789";
40285         if(this.allowDecimals){
40286             allowed += this.decimalSeparator;
40287         }
40288         if(this.allowNegative){
40289             allowed += "-";
40290         }
40291         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40292         var keyPress = function(e){
40293             var k = e.getKey();
40294             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40295                 return;
40296             }
40297             var c = e.getCharCode();
40298             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40299                 e.stopEvent();
40300             }
40301         };
40302         this.el.on("keypress", keyPress, this);
40303     },
40304
40305     // private
40306     validateValue : function(value){
40307         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40308             return false;
40309         }
40310         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40311              return true;
40312         }
40313         var num = this.parseValue(value);
40314         if(isNaN(num)){
40315             this.markInvalid(String.format(this.nanText, value));
40316             return false;
40317         }
40318         if(num < this.minValue){
40319             this.markInvalid(String.format(this.minText, this.minValue));
40320             return false;
40321         }
40322         if(num > this.maxValue){
40323             this.markInvalid(String.format(this.maxText, this.maxValue));
40324             return false;
40325         }
40326         return true;
40327     },
40328
40329     getValue : function(){
40330         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40331     },
40332
40333     // private
40334     parseValue : function(value){
40335         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40336         return isNaN(value) ? '' : value;
40337     },
40338
40339     // private
40340     fixPrecision : function(value){
40341         var nan = isNaN(value);
40342         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40343             return nan ? '' : value;
40344         }
40345         return parseFloat(value).toFixed(this.decimalPrecision);
40346     },
40347
40348     setValue : function(v){
40349         v = this.fixPrecision(v);
40350         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40351     },
40352
40353     // private
40354     decimalPrecisionFcn : function(v){
40355         return Math.floor(v);
40356     },
40357
40358     beforeBlur : function(){
40359         var v = this.parseValue(this.getRawValue());
40360         if(v){
40361             this.setValue(v);
40362         }
40363     }
40364 });/*
40365  * Based on:
40366  * Ext JS Library 1.1.1
40367  * Copyright(c) 2006-2007, Ext JS, LLC.
40368  *
40369  * Originally Released Under LGPL - original licence link has changed is not relivant.
40370  *
40371  * Fork - LGPL
40372  * <script type="text/javascript">
40373  */
40374  
40375 /**
40376  * @class Roo.form.DateField
40377  * @extends Roo.form.TriggerField
40378  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40379 * @constructor
40380 * Create a new DateField
40381 * @param {Object} config
40382  */
40383 Roo.form.DateField = function(config)
40384 {
40385     Roo.form.DateField.superclass.constructor.call(this, config);
40386     
40387       this.addEvents({
40388          
40389         /**
40390          * @event select
40391          * Fires when a date is selected
40392              * @param {Roo.form.DateField} combo This combo box
40393              * @param {Date} date The date selected
40394              */
40395         'select' : true
40396          
40397     });
40398     
40399     
40400     if(typeof this.minValue == "string") {
40401         this.minValue = this.parseDate(this.minValue);
40402     }
40403     if(typeof this.maxValue == "string") {
40404         this.maxValue = this.parseDate(this.maxValue);
40405     }
40406     this.ddMatch = null;
40407     if(this.disabledDates){
40408         var dd = this.disabledDates;
40409         var re = "(?:";
40410         for(var i = 0; i < dd.length; i++){
40411             re += dd[i];
40412             if(i != dd.length-1) {
40413                 re += "|";
40414             }
40415         }
40416         this.ddMatch = new RegExp(re + ")");
40417     }
40418 };
40419
40420 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40421     /**
40422      * @cfg {String} format
40423      * The default date format string which can be overriden for localization support.  The format must be
40424      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40425      */
40426     format : "m/d/y",
40427     /**
40428      * @cfg {String} altFormats
40429      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40430      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40431      */
40432     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40433     /**
40434      * @cfg {Array} disabledDays
40435      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40436      */
40437     disabledDays : null,
40438     /**
40439      * @cfg {String} disabledDaysText
40440      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40441      */
40442     disabledDaysText : "Disabled",
40443     /**
40444      * @cfg {Array} disabledDates
40445      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40446      * expression so they are very powerful. Some examples:
40447      * <ul>
40448      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40449      * <li>["03/08", "09/16"] would disable those days for every year</li>
40450      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40451      * <li>["03/../2006"] would disable every day in March 2006</li>
40452      * <li>["^03"] would disable every day in every March</li>
40453      * </ul>
40454      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40455      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40456      */
40457     disabledDates : null,
40458     /**
40459      * @cfg {String} disabledDatesText
40460      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40461      */
40462     disabledDatesText : "Disabled",
40463     /**
40464      * @cfg {Date/String} minValue
40465      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40466      * valid format (defaults to null).
40467      */
40468     minValue : null,
40469     /**
40470      * @cfg {Date/String} maxValue
40471      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40472      * valid format (defaults to null).
40473      */
40474     maxValue : null,
40475     /**
40476      * @cfg {String} minText
40477      * The error text to display when the date in the cell is before minValue (defaults to
40478      * 'The date in this field must be after {minValue}').
40479      */
40480     minText : "The date in this field must be equal to or after {0}",
40481     /**
40482      * @cfg {String} maxText
40483      * The error text to display when the date in the cell is after maxValue (defaults to
40484      * 'The date in this field must be before {maxValue}').
40485      */
40486     maxText : "The date in this field must be equal to or before {0}",
40487     /**
40488      * @cfg {String} invalidText
40489      * The error text to display when the date in the field is invalid (defaults to
40490      * '{value} is not a valid date - it must be in the format {format}').
40491      */
40492     invalidText : "{0} is not a valid date - it must be in the format {1}",
40493     /**
40494      * @cfg {String} triggerClass
40495      * An additional CSS class used to style the trigger button.  The trigger will always get the
40496      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40497      * which displays a calendar icon).
40498      */
40499     triggerClass : 'x-form-date-trigger',
40500     
40501
40502     /**
40503      * @cfg {Boolean} useIso
40504      * if enabled, then the date field will use a hidden field to store the 
40505      * real value as iso formated date. default (false)
40506      */ 
40507     useIso : false,
40508     /**
40509      * @cfg {String/Object} autoCreate
40510      * A DomHelper element spec, or true for a default element spec (defaults to
40511      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40512      */ 
40513     // private
40514     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40515     
40516     // private
40517     hiddenField: false,
40518     
40519     onRender : function(ct, position)
40520     {
40521         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40522         if (this.useIso) {
40523             //this.el.dom.removeAttribute('name'); 
40524             Roo.log("Changing name?");
40525             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40526             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40527                     'before', true);
40528             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40529             // prevent input submission
40530             this.hiddenName = this.name;
40531         }
40532             
40533             
40534     },
40535     
40536     // private
40537     validateValue : function(value)
40538     {
40539         value = this.formatDate(value);
40540         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40541             Roo.log('super failed');
40542             return false;
40543         }
40544         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40545              return true;
40546         }
40547         var svalue = value;
40548         value = this.parseDate(value);
40549         if(!value){
40550             Roo.log('parse date failed' + svalue);
40551             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40552             return false;
40553         }
40554         var time = value.getTime();
40555         if(this.minValue && time < this.minValue.getTime()){
40556             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40557             return false;
40558         }
40559         if(this.maxValue && time > this.maxValue.getTime()){
40560             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40561             return false;
40562         }
40563         if(this.disabledDays){
40564             var day = value.getDay();
40565             for(var i = 0; i < this.disabledDays.length; i++) {
40566                 if(day === this.disabledDays[i]){
40567                     this.markInvalid(this.disabledDaysText);
40568                     return false;
40569                 }
40570             }
40571         }
40572         var fvalue = this.formatDate(value);
40573         if(this.ddMatch && this.ddMatch.test(fvalue)){
40574             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40575             return false;
40576         }
40577         return true;
40578     },
40579
40580     // private
40581     // Provides logic to override the default TriggerField.validateBlur which just returns true
40582     validateBlur : function(){
40583         return !this.menu || !this.menu.isVisible();
40584     },
40585     
40586     getName: function()
40587     {
40588         // returns hidden if it's set..
40589         if (!this.rendered) {return ''};
40590         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40591         
40592     },
40593
40594     /**
40595      * Returns the current date value of the date field.
40596      * @return {Date} The date value
40597      */
40598     getValue : function(){
40599         
40600         return  this.hiddenField ?
40601                 this.hiddenField.value :
40602                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40603     },
40604
40605     /**
40606      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40607      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40608      * (the default format used is "m/d/y").
40609      * <br />Usage:
40610      * <pre><code>
40611 //All of these calls set the same date value (May 4, 2006)
40612
40613 //Pass a date object:
40614 var dt = new Date('5/4/06');
40615 dateField.setValue(dt);
40616
40617 //Pass a date string (default format):
40618 dateField.setValue('5/4/06');
40619
40620 //Pass a date string (custom format):
40621 dateField.format = 'Y-m-d';
40622 dateField.setValue('2006-5-4');
40623 </code></pre>
40624      * @param {String/Date} date The date or valid date string
40625      */
40626     setValue : function(date){
40627         if (this.hiddenField) {
40628             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40629         }
40630         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40631         // make sure the value field is always stored as a date..
40632         this.value = this.parseDate(date);
40633         
40634         
40635     },
40636
40637     // private
40638     parseDate : function(value){
40639         if(!value || value instanceof Date){
40640             return value;
40641         }
40642         var v = Date.parseDate(value, this.format);
40643          if (!v && this.useIso) {
40644             v = Date.parseDate(value, 'Y-m-d');
40645         }
40646         if(!v && this.altFormats){
40647             if(!this.altFormatsArray){
40648                 this.altFormatsArray = this.altFormats.split("|");
40649             }
40650             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40651                 v = Date.parseDate(value, this.altFormatsArray[i]);
40652             }
40653         }
40654         return v;
40655     },
40656
40657     // private
40658     formatDate : function(date, fmt){
40659         return (!date || !(date instanceof Date)) ?
40660                date : date.dateFormat(fmt || this.format);
40661     },
40662
40663     // private
40664     menuListeners : {
40665         select: function(m, d){
40666             
40667             this.setValue(d);
40668             this.fireEvent('select', this, d);
40669         },
40670         show : function(){ // retain focus styling
40671             this.onFocus();
40672         },
40673         hide : function(){
40674             this.focus.defer(10, this);
40675             var ml = this.menuListeners;
40676             this.menu.un("select", ml.select,  this);
40677             this.menu.un("show", ml.show,  this);
40678             this.menu.un("hide", ml.hide,  this);
40679         }
40680     },
40681
40682     // private
40683     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40684     onTriggerClick : function(){
40685         if(this.disabled){
40686             return;
40687         }
40688         if(this.menu == null){
40689             this.menu = new Roo.menu.DateMenu();
40690         }
40691         Roo.apply(this.menu.picker,  {
40692             showClear: this.allowBlank,
40693             minDate : this.minValue,
40694             maxDate : this.maxValue,
40695             disabledDatesRE : this.ddMatch,
40696             disabledDatesText : this.disabledDatesText,
40697             disabledDays : this.disabledDays,
40698             disabledDaysText : this.disabledDaysText,
40699             format : this.useIso ? 'Y-m-d' : this.format,
40700             minText : String.format(this.minText, this.formatDate(this.minValue)),
40701             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40702         });
40703         this.menu.on(Roo.apply({}, this.menuListeners, {
40704             scope:this
40705         }));
40706         this.menu.picker.setValue(this.getValue() || new Date());
40707         this.menu.show(this.el, "tl-bl?");
40708     },
40709
40710     beforeBlur : function(){
40711         var v = this.parseDate(this.getRawValue());
40712         if(v){
40713             this.setValue(v);
40714         }
40715     },
40716
40717     /*@
40718      * overide
40719      * 
40720      */
40721     isDirty : function() {
40722         if(this.disabled) {
40723             return false;
40724         }
40725         
40726         if(typeof(this.startValue) === 'undefined'){
40727             return false;
40728         }
40729         
40730         return String(this.getValue()) !== String(this.startValue);
40731         
40732     },
40733     // @overide
40734     cleanLeadingSpace : function(e)
40735     {
40736        return;
40737     }
40738     
40739 });/*
40740  * Based on:
40741  * Ext JS Library 1.1.1
40742  * Copyright(c) 2006-2007, Ext JS, LLC.
40743  *
40744  * Originally Released Under LGPL - original licence link has changed is not relivant.
40745  *
40746  * Fork - LGPL
40747  * <script type="text/javascript">
40748  */
40749  
40750 /**
40751  * @class Roo.form.MonthField
40752  * @extends Roo.form.TriggerField
40753  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40754 * @constructor
40755 * Create a new MonthField
40756 * @param {Object} config
40757  */
40758 Roo.form.MonthField = function(config){
40759     
40760     Roo.form.MonthField.superclass.constructor.call(this, config);
40761     
40762       this.addEvents({
40763          
40764         /**
40765          * @event select
40766          * Fires when a date is selected
40767              * @param {Roo.form.MonthFieeld} combo This combo box
40768              * @param {Date} date The date selected
40769              */
40770         'select' : true
40771          
40772     });
40773     
40774     
40775     if(typeof this.minValue == "string") {
40776         this.minValue = this.parseDate(this.minValue);
40777     }
40778     if(typeof this.maxValue == "string") {
40779         this.maxValue = this.parseDate(this.maxValue);
40780     }
40781     this.ddMatch = null;
40782     if(this.disabledDates){
40783         var dd = this.disabledDates;
40784         var re = "(?:";
40785         for(var i = 0; i < dd.length; i++){
40786             re += dd[i];
40787             if(i != dd.length-1) {
40788                 re += "|";
40789             }
40790         }
40791         this.ddMatch = new RegExp(re + ")");
40792     }
40793 };
40794
40795 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40796     /**
40797      * @cfg {String} format
40798      * The default date format string which can be overriden for localization support.  The format must be
40799      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40800      */
40801     format : "M Y",
40802     /**
40803      * @cfg {String} altFormats
40804      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40805      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40806      */
40807     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40808     /**
40809      * @cfg {Array} disabledDays
40810      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40811      */
40812     disabledDays : [0,1,2,3,4,5,6],
40813     /**
40814      * @cfg {String} disabledDaysText
40815      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40816      */
40817     disabledDaysText : "Disabled",
40818     /**
40819      * @cfg {Array} disabledDates
40820      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40821      * expression so they are very powerful. Some examples:
40822      * <ul>
40823      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40824      * <li>["03/08", "09/16"] would disable those days for every year</li>
40825      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40826      * <li>["03/../2006"] would disable every day in March 2006</li>
40827      * <li>["^03"] would disable every day in every March</li>
40828      * </ul>
40829      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40830      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40831      */
40832     disabledDates : null,
40833     /**
40834      * @cfg {String} disabledDatesText
40835      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40836      */
40837     disabledDatesText : "Disabled",
40838     /**
40839      * @cfg {Date/String} minValue
40840      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40841      * valid format (defaults to null).
40842      */
40843     minValue : null,
40844     /**
40845      * @cfg {Date/String} maxValue
40846      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40847      * valid format (defaults to null).
40848      */
40849     maxValue : null,
40850     /**
40851      * @cfg {String} minText
40852      * The error text to display when the date in the cell is before minValue (defaults to
40853      * 'The date in this field must be after {minValue}').
40854      */
40855     minText : "The date in this field must be equal to or after {0}",
40856     /**
40857      * @cfg {String} maxTextf
40858      * The error text to display when the date in the cell is after maxValue (defaults to
40859      * 'The date in this field must be before {maxValue}').
40860      */
40861     maxText : "The date in this field must be equal to or before {0}",
40862     /**
40863      * @cfg {String} invalidText
40864      * The error text to display when the date in the field is invalid (defaults to
40865      * '{value} is not a valid date - it must be in the format {format}').
40866      */
40867     invalidText : "{0} is not a valid date - it must be in the format {1}",
40868     /**
40869      * @cfg {String} triggerClass
40870      * An additional CSS class used to style the trigger button.  The trigger will always get the
40871      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40872      * which displays a calendar icon).
40873      */
40874     triggerClass : 'x-form-date-trigger',
40875     
40876
40877     /**
40878      * @cfg {Boolean} useIso
40879      * if enabled, then the date field will use a hidden field to store the 
40880      * real value as iso formated date. default (true)
40881      */ 
40882     useIso : true,
40883     /**
40884      * @cfg {String/Object} autoCreate
40885      * A DomHelper element spec, or true for a default element spec (defaults to
40886      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40887      */ 
40888     // private
40889     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40890     
40891     // private
40892     hiddenField: false,
40893     
40894     hideMonthPicker : false,
40895     
40896     onRender : function(ct, position)
40897     {
40898         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40899         if (this.useIso) {
40900             this.el.dom.removeAttribute('name'); 
40901             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40902                     'before', true);
40903             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40904             // prevent input submission
40905             this.hiddenName = this.name;
40906         }
40907             
40908             
40909     },
40910     
40911     // private
40912     validateValue : function(value)
40913     {
40914         value = this.formatDate(value);
40915         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40916             return false;
40917         }
40918         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40919              return true;
40920         }
40921         var svalue = value;
40922         value = this.parseDate(value);
40923         if(!value){
40924             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40925             return false;
40926         }
40927         var time = value.getTime();
40928         if(this.minValue && time < this.minValue.getTime()){
40929             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40930             return false;
40931         }
40932         if(this.maxValue && time > this.maxValue.getTime()){
40933             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40934             return false;
40935         }
40936         /*if(this.disabledDays){
40937             var day = value.getDay();
40938             for(var i = 0; i < this.disabledDays.length; i++) {
40939                 if(day === this.disabledDays[i]){
40940                     this.markInvalid(this.disabledDaysText);
40941                     return false;
40942                 }
40943             }
40944         }
40945         */
40946         var fvalue = this.formatDate(value);
40947         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40948             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40949             return false;
40950         }
40951         */
40952         return true;
40953     },
40954
40955     // private
40956     // Provides logic to override the default TriggerField.validateBlur which just returns true
40957     validateBlur : function(){
40958         return !this.menu || !this.menu.isVisible();
40959     },
40960
40961     /**
40962      * Returns the current date value of the date field.
40963      * @return {Date} The date value
40964      */
40965     getValue : function(){
40966         
40967         
40968         
40969         return  this.hiddenField ?
40970                 this.hiddenField.value :
40971                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40972     },
40973
40974     /**
40975      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40976      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40977      * (the default format used is "m/d/y").
40978      * <br />Usage:
40979      * <pre><code>
40980 //All of these calls set the same date value (May 4, 2006)
40981
40982 //Pass a date object:
40983 var dt = new Date('5/4/06');
40984 monthField.setValue(dt);
40985
40986 //Pass a date string (default format):
40987 monthField.setValue('5/4/06');
40988
40989 //Pass a date string (custom format):
40990 monthField.format = 'Y-m-d';
40991 monthField.setValue('2006-5-4');
40992 </code></pre>
40993      * @param {String/Date} date The date or valid date string
40994      */
40995     setValue : function(date){
40996         Roo.log('month setValue' + date);
40997         // can only be first of month..
40998         
40999         var val = this.parseDate(date);
41000         
41001         if (this.hiddenField) {
41002             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
41003         }
41004         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
41005         this.value = this.parseDate(date);
41006     },
41007
41008     // private
41009     parseDate : function(value){
41010         if(!value || value instanceof Date){
41011             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
41012             return value;
41013         }
41014         var v = Date.parseDate(value, this.format);
41015         if (!v && this.useIso) {
41016             v = Date.parseDate(value, 'Y-m-d');
41017         }
41018         if (v) {
41019             // 
41020             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
41021         }
41022         
41023         
41024         if(!v && this.altFormats){
41025             if(!this.altFormatsArray){
41026                 this.altFormatsArray = this.altFormats.split("|");
41027             }
41028             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
41029                 v = Date.parseDate(value, this.altFormatsArray[i]);
41030             }
41031         }
41032         return v;
41033     },
41034
41035     // private
41036     formatDate : function(date, fmt){
41037         return (!date || !(date instanceof Date)) ?
41038                date : date.dateFormat(fmt || this.format);
41039     },
41040
41041     // private
41042     menuListeners : {
41043         select: function(m, d){
41044             this.setValue(d);
41045             this.fireEvent('select', this, d);
41046         },
41047         show : function(){ // retain focus styling
41048             this.onFocus();
41049         },
41050         hide : function(){
41051             this.focus.defer(10, this);
41052             var ml = this.menuListeners;
41053             this.menu.un("select", ml.select,  this);
41054             this.menu.un("show", ml.show,  this);
41055             this.menu.un("hide", ml.hide,  this);
41056         }
41057     },
41058     // private
41059     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41060     onTriggerClick : function(){
41061         if(this.disabled){
41062             return;
41063         }
41064         if(this.menu == null){
41065             this.menu = new Roo.menu.DateMenu();
41066            
41067         }
41068         
41069         Roo.apply(this.menu.picker,  {
41070             
41071             showClear: this.allowBlank,
41072             minDate : this.minValue,
41073             maxDate : this.maxValue,
41074             disabledDatesRE : this.ddMatch,
41075             disabledDatesText : this.disabledDatesText,
41076             
41077             format : this.useIso ? 'Y-m-d' : this.format,
41078             minText : String.format(this.minText, this.formatDate(this.minValue)),
41079             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41080             
41081         });
41082          this.menu.on(Roo.apply({}, this.menuListeners, {
41083             scope:this
41084         }));
41085        
41086         
41087         var m = this.menu;
41088         var p = m.picker;
41089         
41090         // hide month picker get's called when we called by 'before hide';
41091         
41092         var ignorehide = true;
41093         p.hideMonthPicker  = function(disableAnim){
41094             if (ignorehide) {
41095                 return;
41096             }
41097              if(this.monthPicker){
41098                 Roo.log("hideMonthPicker called");
41099                 if(disableAnim === true){
41100                     this.monthPicker.hide();
41101                 }else{
41102                     this.monthPicker.slideOut('t', {duration:.2});
41103                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41104                     p.fireEvent("select", this, this.value);
41105                     m.hide();
41106                 }
41107             }
41108         }
41109         
41110         Roo.log('picker set value');
41111         Roo.log(this.getValue());
41112         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41113         m.show(this.el, 'tl-bl?');
41114         ignorehide  = false;
41115         // this will trigger hideMonthPicker..
41116         
41117         
41118         // hidden the day picker
41119         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41120         
41121         
41122         
41123       
41124         
41125         p.showMonthPicker.defer(100, p);
41126     
41127         
41128        
41129     },
41130
41131     beforeBlur : function(){
41132         var v = this.parseDate(this.getRawValue());
41133         if(v){
41134             this.setValue(v);
41135         }
41136     }
41137
41138     /** @cfg {Boolean} grow @hide */
41139     /** @cfg {Number} growMin @hide */
41140     /** @cfg {Number} growMax @hide */
41141     /**
41142      * @hide
41143      * @method autoSize
41144      */
41145 });/*
41146  * Based on:
41147  * Ext JS Library 1.1.1
41148  * Copyright(c) 2006-2007, Ext JS, LLC.
41149  *
41150  * Originally Released Under LGPL - original licence link has changed is not relivant.
41151  *
41152  * Fork - LGPL
41153  * <script type="text/javascript">
41154  */
41155  
41156
41157 /**
41158  * @class Roo.form.ComboBox
41159  * @extends Roo.form.TriggerField
41160  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41161  * @constructor
41162  * Create a new ComboBox.
41163  * @param {Object} config Configuration options
41164  */
41165 Roo.form.ComboBox = function(config){
41166     Roo.form.ComboBox.superclass.constructor.call(this, config);
41167     this.addEvents({
41168         /**
41169          * @event expand
41170          * Fires when the dropdown list is expanded
41171              * @param {Roo.form.ComboBox} combo This combo box
41172              */
41173         'expand' : true,
41174         /**
41175          * @event collapse
41176          * Fires when the dropdown list is collapsed
41177              * @param {Roo.form.ComboBox} combo This combo box
41178              */
41179         'collapse' : true,
41180         /**
41181          * @event beforeselect
41182          * Fires before a list item is selected. Return false to cancel the selection.
41183              * @param {Roo.form.ComboBox} combo This combo box
41184              * @param {Roo.data.Record} record The data record returned from the underlying store
41185              * @param {Number} index The index of the selected item in the dropdown list
41186              */
41187         'beforeselect' : true,
41188         /**
41189          * @event select
41190          * Fires when a list item is selected
41191              * @param {Roo.form.ComboBox} combo This combo box
41192              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41193              * @param {Number} index The index of the selected item in the dropdown list
41194              */
41195         'select' : true,
41196         /**
41197          * @event beforequery
41198          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41199          * The event object passed has these properties:
41200              * @param {Roo.form.ComboBox} combo This combo box
41201              * @param {String} query The query
41202              * @param {Boolean} forceAll true to force "all" query
41203              * @param {Boolean} cancel true to cancel the query
41204              * @param {Object} e The query event object
41205              */
41206         'beforequery': true,
41207          /**
41208          * @event add
41209          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41210              * @param {Roo.form.ComboBox} combo This combo box
41211              */
41212         'add' : true,
41213         /**
41214          * @event edit
41215          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41216              * @param {Roo.form.ComboBox} combo This combo box
41217              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41218              */
41219         'edit' : true
41220         
41221         
41222     });
41223     if(this.transform){
41224         this.allowDomMove = false;
41225         var s = Roo.getDom(this.transform);
41226         if(!this.hiddenName){
41227             this.hiddenName = s.name;
41228         }
41229         if(!this.store){
41230             this.mode = 'local';
41231             var d = [], opts = s.options;
41232             for(var i = 0, len = opts.length;i < len; i++){
41233                 var o = opts[i];
41234                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41235                 if(o.selected) {
41236                     this.value = value;
41237                 }
41238                 d.push([value, o.text]);
41239             }
41240             this.store = new Roo.data.SimpleStore({
41241                 'id': 0,
41242                 fields: ['value', 'text'],
41243                 data : d
41244             });
41245             this.valueField = 'value';
41246             this.displayField = 'text';
41247         }
41248         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41249         if(!this.lazyRender){
41250             this.target = true;
41251             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41252             s.parentNode.removeChild(s); // remove it
41253             this.render(this.el.parentNode);
41254         }else{
41255             s.parentNode.removeChild(s); // remove it
41256         }
41257
41258     }
41259     if (this.store) {
41260         this.store = Roo.factory(this.store, Roo.data);
41261     }
41262     
41263     this.selectedIndex = -1;
41264     if(this.mode == 'local'){
41265         if(config.queryDelay === undefined){
41266             this.queryDelay = 10;
41267         }
41268         if(config.minChars === undefined){
41269             this.minChars = 0;
41270         }
41271     }
41272 };
41273
41274 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41275     /**
41276      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41277      */
41278     /**
41279      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41280      * rendering into an Roo.Editor, defaults to false)
41281      */
41282     /**
41283      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41284      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41285      */
41286     /**
41287      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41288      */
41289     /**
41290      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41291      * the dropdown list (defaults to undefined, with no header element)
41292      */
41293
41294      /**
41295      * @cfg {String/Roo.Template} tpl The template to use to render the output
41296      */
41297      
41298     // private
41299     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41300     /**
41301      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41302      */
41303     listWidth: undefined,
41304     /**
41305      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41306      * mode = 'remote' or 'text' if mode = 'local')
41307      */
41308     displayField: undefined,
41309     /**
41310      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41311      * mode = 'remote' or 'value' if mode = 'local'). 
41312      * Note: use of a valueField requires the user make a selection
41313      * in order for a value to be mapped.
41314      */
41315     valueField: undefined,
41316     
41317     
41318     /**
41319      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41320      * field's data value (defaults to the underlying DOM element's name)
41321      */
41322     hiddenName: undefined,
41323     /**
41324      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41325      */
41326     listClass: '',
41327     /**
41328      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41329      */
41330     selectedClass: 'x-combo-selected',
41331     /**
41332      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41333      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41334      * which displays a downward arrow icon).
41335      */
41336     triggerClass : 'x-form-arrow-trigger',
41337     /**
41338      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41339      */
41340     shadow:'sides',
41341     /**
41342      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41343      * anchor positions (defaults to 'tl-bl')
41344      */
41345     listAlign: 'tl-bl?',
41346     /**
41347      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41348      */
41349     maxHeight: 300,
41350     /**
41351      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41352      * query specified by the allQuery config option (defaults to 'query')
41353      */
41354     triggerAction: 'query',
41355     /**
41356      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41357      * (defaults to 4, does not apply if editable = false)
41358      */
41359     minChars : 4,
41360     /**
41361      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41362      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41363      */
41364     typeAhead: false,
41365     /**
41366      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41367      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41368      */
41369     queryDelay: 500,
41370     /**
41371      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41372      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41373      */
41374     pageSize: 0,
41375     /**
41376      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41377      * when editable = true (defaults to false)
41378      */
41379     selectOnFocus:false,
41380     /**
41381      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41382      */
41383     queryParam: 'query',
41384     /**
41385      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41386      * when mode = 'remote' (defaults to 'Loading...')
41387      */
41388     loadingText: 'Loading...',
41389     /**
41390      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41391      */
41392     resizable: false,
41393     /**
41394      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41395      */
41396     handleHeight : 8,
41397     /**
41398      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41399      * traditional select (defaults to true)
41400      */
41401     editable: true,
41402     /**
41403      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41404      */
41405     allQuery: '',
41406     /**
41407      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41408      */
41409     mode: 'remote',
41410     /**
41411      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41412      * listWidth has a higher value)
41413      */
41414     minListWidth : 70,
41415     /**
41416      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41417      * allow the user to set arbitrary text into the field (defaults to false)
41418      */
41419     forceSelection:false,
41420     /**
41421      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41422      * if typeAhead = true (defaults to 250)
41423      */
41424     typeAheadDelay : 250,
41425     /**
41426      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41427      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41428      */
41429     valueNotFoundText : undefined,
41430     /**
41431      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41432      */
41433     blockFocus : false,
41434     
41435     /**
41436      * @cfg {Boolean} disableClear Disable showing of clear button.
41437      */
41438     disableClear : false,
41439     /**
41440      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41441      */
41442     alwaysQuery : false,
41443     
41444     //private
41445     addicon : false,
41446     editicon: false,
41447     
41448     // element that contains real text value.. (when hidden is used..)
41449      
41450     // private
41451     onRender : function(ct, position)
41452     {
41453         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41454         
41455         if(this.hiddenName){
41456             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41457                     'before', true);
41458             this.hiddenField.value =
41459                 this.hiddenValue !== undefined ? this.hiddenValue :
41460                 this.value !== undefined ? this.value : '';
41461
41462             // prevent input submission
41463             this.el.dom.removeAttribute('name');
41464              
41465              
41466         }
41467         
41468         if(Roo.isGecko){
41469             this.el.dom.setAttribute('autocomplete', 'off');
41470         }
41471
41472         var cls = 'x-combo-list';
41473
41474         this.list = new Roo.Layer({
41475             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41476         });
41477
41478         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41479         this.list.setWidth(lw);
41480         this.list.swallowEvent('mousewheel');
41481         this.assetHeight = 0;
41482
41483         if(this.title){
41484             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41485             this.assetHeight += this.header.getHeight();
41486         }
41487
41488         this.innerList = this.list.createChild({cls:cls+'-inner'});
41489         this.innerList.on('mouseover', this.onViewOver, this);
41490         this.innerList.on('mousemove', this.onViewMove, this);
41491         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41492         
41493         if(this.allowBlank && !this.pageSize && !this.disableClear){
41494             this.footer = this.list.createChild({cls:cls+'-ft'});
41495             this.pageTb = new Roo.Toolbar(this.footer);
41496            
41497         }
41498         if(this.pageSize){
41499             this.footer = this.list.createChild({cls:cls+'-ft'});
41500             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41501                     {pageSize: this.pageSize});
41502             
41503         }
41504         
41505         if (this.pageTb && this.allowBlank && !this.disableClear) {
41506             var _this = this;
41507             this.pageTb.add(new Roo.Toolbar.Fill(), {
41508                 cls: 'x-btn-icon x-btn-clear',
41509                 text: '&#160;',
41510                 handler: function()
41511                 {
41512                     _this.collapse();
41513                     _this.clearValue();
41514                     _this.onSelect(false, -1);
41515                 }
41516             });
41517         }
41518         if (this.footer) {
41519             this.assetHeight += this.footer.getHeight();
41520         }
41521         
41522
41523         if(!this.tpl){
41524             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41525         }
41526
41527         this.view = new Roo.View(this.innerList, this.tpl, {
41528             singleSelect:true,
41529             store: this.store,
41530             selectedClass: this.selectedClass
41531         });
41532
41533         this.view.on('click', this.onViewClick, this);
41534
41535         this.store.on('beforeload', this.onBeforeLoad, this);
41536         this.store.on('load', this.onLoad, this);
41537         this.store.on('loadexception', this.onLoadException, this);
41538
41539         if(this.resizable){
41540             this.resizer = new Roo.Resizable(this.list,  {
41541                pinned:true, handles:'se'
41542             });
41543             this.resizer.on('resize', function(r, w, h){
41544                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41545                 this.listWidth = w;
41546                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41547                 this.restrictHeight();
41548             }, this);
41549             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41550         }
41551         if(!this.editable){
41552             this.editable = true;
41553             this.setEditable(false);
41554         }  
41555         
41556         
41557         if (typeof(this.events.add.listeners) != 'undefined') {
41558             
41559             this.addicon = this.wrap.createChild(
41560                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41561        
41562             this.addicon.on('click', function(e) {
41563                 this.fireEvent('add', this);
41564             }, this);
41565         }
41566         if (typeof(this.events.edit.listeners) != 'undefined') {
41567             
41568             this.editicon = this.wrap.createChild(
41569                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41570             if (this.addicon) {
41571                 this.editicon.setStyle('margin-left', '40px');
41572             }
41573             this.editicon.on('click', function(e) {
41574                 
41575                 // we fire even  if inothing is selected..
41576                 this.fireEvent('edit', this, this.lastData );
41577                 
41578             }, this);
41579         }
41580         
41581         
41582         
41583     },
41584
41585     // private
41586     initEvents : function(){
41587         Roo.form.ComboBox.superclass.initEvents.call(this);
41588
41589         this.keyNav = new Roo.KeyNav(this.el, {
41590             "up" : function(e){
41591                 this.inKeyMode = true;
41592                 this.selectPrev();
41593             },
41594
41595             "down" : function(e){
41596                 if(!this.isExpanded()){
41597                     this.onTriggerClick();
41598                 }else{
41599                     this.inKeyMode = true;
41600                     this.selectNext();
41601                 }
41602             },
41603
41604             "enter" : function(e){
41605                 this.onViewClick();
41606                 //return true;
41607             },
41608
41609             "esc" : function(e){
41610                 this.collapse();
41611             },
41612
41613             "tab" : function(e){
41614                 this.onViewClick(false);
41615                 this.fireEvent("specialkey", this, e);
41616                 return true;
41617             },
41618
41619             scope : this,
41620
41621             doRelay : function(foo, bar, hname){
41622                 if(hname == 'down' || this.scope.isExpanded()){
41623                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41624                 }
41625                 return true;
41626             },
41627
41628             forceKeyDown: true
41629         });
41630         this.queryDelay = Math.max(this.queryDelay || 10,
41631                 this.mode == 'local' ? 10 : 250);
41632         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41633         if(this.typeAhead){
41634             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41635         }
41636         if(this.editable !== false){
41637             this.el.on("keyup", this.onKeyUp, this);
41638         }
41639         if(this.forceSelection){
41640             this.on('blur', this.doForce, this);
41641         }
41642     },
41643
41644     onDestroy : function(){
41645         if(this.view){
41646             this.view.setStore(null);
41647             this.view.el.removeAllListeners();
41648             this.view.el.remove();
41649             this.view.purgeListeners();
41650         }
41651         if(this.list){
41652             this.list.destroy();
41653         }
41654         if(this.store){
41655             this.store.un('beforeload', this.onBeforeLoad, this);
41656             this.store.un('load', this.onLoad, this);
41657             this.store.un('loadexception', this.onLoadException, this);
41658         }
41659         Roo.form.ComboBox.superclass.onDestroy.call(this);
41660     },
41661
41662     // private
41663     fireKey : function(e){
41664         if(e.isNavKeyPress() && !this.list.isVisible()){
41665             this.fireEvent("specialkey", this, e);
41666         }
41667     },
41668
41669     // private
41670     onResize: function(w, h){
41671         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41672         
41673         if(typeof w != 'number'){
41674             // we do not handle it!?!?
41675             return;
41676         }
41677         var tw = this.trigger.getWidth();
41678         tw += this.addicon ? this.addicon.getWidth() : 0;
41679         tw += this.editicon ? this.editicon.getWidth() : 0;
41680         var x = w - tw;
41681         this.el.setWidth( this.adjustWidth('input', x));
41682             
41683         this.trigger.setStyle('left', x+'px');
41684         
41685         if(this.list && this.listWidth === undefined){
41686             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41687             this.list.setWidth(lw);
41688             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41689         }
41690         
41691     
41692         
41693     },
41694
41695     /**
41696      * Allow or prevent the user from directly editing the field text.  If false is passed,
41697      * the user will only be able to select from the items defined in the dropdown list.  This method
41698      * is the runtime equivalent of setting the 'editable' config option at config time.
41699      * @param {Boolean} value True to allow the user to directly edit the field text
41700      */
41701     setEditable : function(value){
41702         if(value == this.editable){
41703             return;
41704         }
41705         this.editable = value;
41706         if(!value){
41707             this.el.dom.setAttribute('readOnly', true);
41708             this.el.on('mousedown', this.onTriggerClick,  this);
41709             this.el.addClass('x-combo-noedit');
41710         }else{
41711             this.el.dom.setAttribute('readOnly', false);
41712             this.el.un('mousedown', this.onTriggerClick,  this);
41713             this.el.removeClass('x-combo-noedit');
41714         }
41715     },
41716
41717     // private
41718     onBeforeLoad : function(){
41719         if(!this.hasFocus){
41720             return;
41721         }
41722         this.innerList.update(this.loadingText ?
41723                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41724         this.restrictHeight();
41725         this.selectedIndex = -1;
41726     },
41727
41728     // private
41729     onLoad : function(){
41730         if(!this.hasFocus){
41731             return;
41732         }
41733         if(this.store.getCount() > 0){
41734             this.expand();
41735             this.restrictHeight();
41736             if(this.lastQuery == this.allQuery){
41737                 if(this.editable){
41738                     this.el.dom.select();
41739                 }
41740                 if(!this.selectByValue(this.value, true)){
41741                     this.select(0, true);
41742                 }
41743             }else{
41744                 this.selectNext();
41745                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41746                     this.taTask.delay(this.typeAheadDelay);
41747                 }
41748             }
41749         }else{
41750             this.onEmptyResults();
41751         }
41752         //this.el.focus();
41753     },
41754     // private
41755     onLoadException : function()
41756     {
41757         this.collapse();
41758         Roo.log(this.store.reader.jsonData);
41759         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41760             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41761         }
41762         
41763         
41764     },
41765     // private
41766     onTypeAhead : function(){
41767         if(this.store.getCount() > 0){
41768             var r = this.store.getAt(0);
41769             var newValue = r.data[this.displayField];
41770             var len = newValue.length;
41771             var selStart = this.getRawValue().length;
41772             if(selStart != len){
41773                 this.setRawValue(newValue);
41774                 this.selectText(selStart, newValue.length);
41775             }
41776         }
41777     },
41778
41779     // private
41780     onSelect : function(record, index){
41781         if(this.fireEvent('beforeselect', this, record, index) !== false){
41782             this.setFromData(index > -1 ? record.data : false);
41783             this.collapse();
41784             this.fireEvent('select', this, record, index);
41785         }
41786     },
41787
41788     /**
41789      * Returns the currently selected field value or empty string if no value is set.
41790      * @return {String} value The selected value
41791      */
41792     getValue : function(){
41793         if(this.valueField){
41794             return typeof this.value != 'undefined' ? this.value : '';
41795         }
41796         return Roo.form.ComboBox.superclass.getValue.call(this);
41797     },
41798
41799     /**
41800      * Clears any text/value currently set in the field
41801      */
41802     clearValue : function(){
41803         if(this.hiddenField){
41804             this.hiddenField.value = '';
41805         }
41806         this.value = '';
41807         this.setRawValue('');
41808         this.lastSelectionText = '';
41809         
41810     },
41811
41812     /**
41813      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41814      * will be displayed in the field.  If the value does not match the data value of an existing item,
41815      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41816      * Otherwise the field will be blank (although the value will still be set).
41817      * @param {String} value The value to match
41818      */
41819     setValue : function(v){
41820         var text = v;
41821         if(this.valueField){
41822             var r = this.findRecord(this.valueField, v);
41823             if(r){
41824                 text = r.data[this.displayField];
41825             }else if(this.valueNotFoundText !== undefined){
41826                 text = this.valueNotFoundText;
41827             }
41828         }
41829         this.lastSelectionText = text;
41830         if(this.hiddenField){
41831             this.hiddenField.value = v;
41832         }
41833         Roo.form.ComboBox.superclass.setValue.call(this, text);
41834         this.value = v;
41835     },
41836     /**
41837      * @property {Object} the last set data for the element
41838      */
41839     
41840     lastData : false,
41841     /**
41842      * Sets the value of the field based on a object which is related to the record format for the store.
41843      * @param {Object} value the value to set as. or false on reset?
41844      */
41845     setFromData : function(o){
41846         var dv = ''; // display value
41847         var vv = ''; // value value..
41848         this.lastData = o;
41849         if (this.displayField) {
41850             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41851         } else {
41852             // this is an error condition!!!
41853             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41854         }
41855         
41856         if(this.valueField){
41857             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41858         }
41859         if(this.hiddenField){
41860             this.hiddenField.value = vv;
41861             
41862             this.lastSelectionText = dv;
41863             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41864             this.value = vv;
41865             return;
41866         }
41867         // no hidden field.. - we store the value in 'value', but still display
41868         // display field!!!!
41869         this.lastSelectionText = dv;
41870         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41871         this.value = vv;
41872         
41873         
41874     },
41875     // private
41876     reset : function(){
41877         // overridden so that last data is reset..
41878         this.setValue(this.resetValue);
41879         this.originalValue = this.getValue();
41880         this.clearInvalid();
41881         this.lastData = false;
41882         if (this.view) {
41883             this.view.clearSelections();
41884         }
41885     },
41886     // private
41887     findRecord : function(prop, value){
41888         var record;
41889         if(this.store.getCount() > 0){
41890             this.store.each(function(r){
41891                 if(r.data[prop] == value){
41892                     record = r;
41893                     return false;
41894                 }
41895                 return true;
41896             });
41897         }
41898         return record;
41899     },
41900     
41901     getName: function()
41902     {
41903         // returns hidden if it's set..
41904         if (!this.rendered) {return ''};
41905         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41906         
41907     },
41908     // private
41909     onViewMove : function(e, t){
41910         this.inKeyMode = false;
41911     },
41912
41913     // private
41914     onViewOver : function(e, t){
41915         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41916             return;
41917         }
41918         var item = this.view.findItemFromChild(t);
41919         if(item){
41920             var index = this.view.indexOf(item);
41921             this.select(index, false);
41922         }
41923     },
41924
41925     // private
41926     onViewClick : function(doFocus)
41927     {
41928         var index = this.view.getSelectedIndexes()[0];
41929         var r = this.store.getAt(index);
41930         if(r){
41931             this.onSelect(r, index);
41932         }
41933         if(doFocus !== false && !this.blockFocus){
41934             this.el.focus();
41935         }
41936     },
41937
41938     // private
41939     restrictHeight : function(){
41940         this.innerList.dom.style.height = '';
41941         var inner = this.innerList.dom;
41942         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41943         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41944         this.list.beginUpdate();
41945         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41946         this.list.alignTo(this.el, this.listAlign);
41947         this.list.endUpdate();
41948     },
41949
41950     // private
41951     onEmptyResults : function(){
41952         this.collapse();
41953     },
41954
41955     /**
41956      * Returns true if the dropdown list is expanded, else false.
41957      */
41958     isExpanded : function(){
41959         return this.list.isVisible();
41960     },
41961
41962     /**
41963      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41964      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41965      * @param {String} value The data value of the item to select
41966      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41967      * selected item if it is not currently in view (defaults to true)
41968      * @return {Boolean} True if the value matched an item in the list, else false
41969      */
41970     selectByValue : function(v, scrollIntoView){
41971         if(v !== undefined && v !== null){
41972             var r = this.findRecord(this.valueField || this.displayField, v);
41973             if(r){
41974                 this.select(this.store.indexOf(r), scrollIntoView);
41975                 return true;
41976             }
41977         }
41978         return false;
41979     },
41980
41981     /**
41982      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41983      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41984      * @param {Number} index The zero-based index of the list item to select
41985      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41986      * selected item if it is not currently in view (defaults to true)
41987      */
41988     select : function(index, scrollIntoView){
41989         this.selectedIndex = index;
41990         this.view.select(index);
41991         if(scrollIntoView !== false){
41992             var el = this.view.getNode(index);
41993             if(el){
41994                 this.innerList.scrollChildIntoView(el, false);
41995             }
41996         }
41997     },
41998
41999     // private
42000     selectNext : function(){
42001         var ct = this.store.getCount();
42002         if(ct > 0){
42003             if(this.selectedIndex == -1){
42004                 this.select(0);
42005             }else if(this.selectedIndex < ct-1){
42006                 this.select(this.selectedIndex+1);
42007             }
42008         }
42009     },
42010
42011     // private
42012     selectPrev : function(){
42013         var ct = this.store.getCount();
42014         if(ct > 0){
42015             if(this.selectedIndex == -1){
42016                 this.select(0);
42017             }else if(this.selectedIndex != 0){
42018                 this.select(this.selectedIndex-1);
42019             }
42020         }
42021     },
42022
42023     // private
42024     onKeyUp : function(e){
42025         if(this.editable !== false && !e.isSpecialKey()){
42026             this.lastKey = e.getKey();
42027             this.dqTask.delay(this.queryDelay);
42028         }
42029     },
42030
42031     // private
42032     validateBlur : function(){
42033         return !this.list || !this.list.isVisible();   
42034     },
42035
42036     // private
42037     initQuery : function(){
42038         this.doQuery(this.getRawValue());
42039     },
42040
42041     // private
42042     doForce : function(){
42043         if(this.el.dom.value.length > 0){
42044             this.el.dom.value =
42045                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
42046              
42047         }
42048     },
42049
42050     /**
42051      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
42052      * query allowing the query action to be canceled if needed.
42053      * @param {String} query The SQL query to execute
42054      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42055      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42056      * saved in the current store (defaults to false)
42057      */
42058     doQuery : function(q, forceAll){
42059         if(q === undefined || q === null){
42060             q = '';
42061         }
42062         var qe = {
42063             query: q,
42064             forceAll: forceAll,
42065             combo: this,
42066             cancel:false
42067         };
42068         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42069             return false;
42070         }
42071         q = qe.query;
42072         forceAll = qe.forceAll;
42073         if(forceAll === true || (q.length >= this.minChars)){
42074             if(this.lastQuery != q || this.alwaysQuery){
42075                 this.lastQuery = q;
42076                 if(this.mode == 'local'){
42077                     this.selectedIndex = -1;
42078                     if(forceAll){
42079                         this.store.clearFilter();
42080                     }else{
42081                         this.store.filter(this.displayField, q);
42082                     }
42083                     this.onLoad();
42084                 }else{
42085                     this.store.baseParams[this.queryParam] = q;
42086                     this.store.load({
42087                         params: this.getParams(q)
42088                     });
42089                     this.expand();
42090                 }
42091             }else{
42092                 this.selectedIndex = -1;
42093                 this.onLoad();   
42094             }
42095         }
42096     },
42097
42098     // private
42099     getParams : function(q){
42100         var p = {};
42101         //p[this.queryParam] = q;
42102         if(this.pageSize){
42103             p.start = 0;
42104             p.limit = this.pageSize;
42105         }
42106         return p;
42107     },
42108
42109     /**
42110      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42111      */
42112     collapse : function(){
42113         if(!this.isExpanded()){
42114             return;
42115         }
42116         this.list.hide();
42117         Roo.get(document).un('mousedown', this.collapseIf, this);
42118         Roo.get(document).un('mousewheel', this.collapseIf, this);
42119         if (!this.editable) {
42120             Roo.get(document).un('keydown', this.listKeyPress, this);
42121         }
42122         this.fireEvent('collapse', this);
42123     },
42124
42125     // private
42126     collapseIf : function(e){
42127         if(!e.within(this.wrap) && !e.within(this.list)){
42128             this.collapse();
42129         }
42130     },
42131
42132     /**
42133      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42134      */
42135     expand : function(){
42136         if(this.isExpanded() || !this.hasFocus){
42137             return;
42138         }
42139         this.list.alignTo(this.el, this.listAlign);
42140         this.list.show();
42141         Roo.get(document).on('mousedown', this.collapseIf, this);
42142         Roo.get(document).on('mousewheel', this.collapseIf, this);
42143         if (!this.editable) {
42144             Roo.get(document).on('keydown', this.listKeyPress, this);
42145         }
42146         
42147         this.fireEvent('expand', this);
42148     },
42149
42150     // private
42151     // Implements the default empty TriggerField.onTriggerClick function
42152     onTriggerClick : function(){
42153         if(this.disabled){
42154             return;
42155         }
42156         if(this.isExpanded()){
42157             this.collapse();
42158             if (!this.blockFocus) {
42159                 this.el.focus();
42160             }
42161             
42162         }else {
42163             this.hasFocus = true;
42164             if(this.triggerAction == 'all') {
42165                 this.doQuery(this.allQuery, true);
42166             } else {
42167                 this.doQuery(this.getRawValue());
42168             }
42169             if (!this.blockFocus) {
42170                 this.el.focus();
42171             }
42172         }
42173     },
42174     listKeyPress : function(e)
42175     {
42176         //Roo.log('listkeypress');
42177         // scroll to first matching element based on key pres..
42178         if (e.isSpecialKey()) {
42179             return false;
42180         }
42181         var k = String.fromCharCode(e.getKey()).toUpperCase();
42182         //Roo.log(k);
42183         var match  = false;
42184         var csel = this.view.getSelectedNodes();
42185         var cselitem = false;
42186         if (csel.length) {
42187             var ix = this.view.indexOf(csel[0]);
42188             cselitem  = this.store.getAt(ix);
42189             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42190                 cselitem = false;
42191             }
42192             
42193         }
42194         
42195         this.store.each(function(v) { 
42196             if (cselitem) {
42197                 // start at existing selection.
42198                 if (cselitem.id == v.id) {
42199                     cselitem = false;
42200                 }
42201                 return;
42202             }
42203                 
42204             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42205                 match = this.store.indexOf(v);
42206                 return false;
42207             }
42208         }, this);
42209         
42210         if (match === false) {
42211             return true; // no more action?
42212         }
42213         // scroll to?
42214         this.view.select(match);
42215         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42216         sn.scrollIntoView(sn.dom.parentNode, false);
42217     } 
42218
42219     /** 
42220     * @cfg {Boolean} grow 
42221     * @hide 
42222     */
42223     /** 
42224     * @cfg {Number} growMin 
42225     * @hide 
42226     */
42227     /** 
42228     * @cfg {Number} growMax 
42229     * @hide 
42230     */
42231     /**
42232      * @hide
42233      * @method autoSize
42234      */
42235 });/*
42236  * Copyright(c) 2010-2012, Roo J Solutions Limited
42237  *
42238  * Licence LGPL
42239  *
42240  */
42241
42242 /**
42243  * @class Roo.form.ComboBoxArray
42244  * @extends Roo.form.TextField
42245  * A facebook style adder... for lists of email / people / countries  etc...
42246  * pick multiple items from a combo box, and shows each one.
42247  *
42248  *  Fred [x]  Brian [x]  [Pick another |v]
42249  *
42250  *
42251  *  For this to work: it needs various extra information
42252  *    - normal combo problay has
42253  *      name, hiddenName
42254  *    + displayField, valueField
42255  *
42256  *    For our purpose...
42257  *
42258  *
42259  *   If we change from 'extends' to wrapping...
42260  *   
42261  *  
42262  *
42263  
42264  
42265  * @constructor
42266  * Create a new ComboBoxArray.
42267  * @param {Object} config Configuration options
42268  */
42269  
42270
42271 Roo.form.ComboBoxArray = function(config)
42272 {
42273     this.addEvents({
42274         /**
42275          * @event beforeremove
42276          * Fires before remove the value from the list
42277              * @param {Roo.form.ComboBoxArray} _self This combo box array
42278              * @param {Roo.form.ComboBoxArray.Item} item removed item
42279              */
42280         'beforeremove' : true,
42281         /**
42282          * @event remove
42283          * Fires when remove the value from the list
42284              * @param {Roo.form.ComboBoxArray} _self This combo box array
42285              * @param {Roo.form.ComboBoxArray.Item} item removed item
42286              */
42287         'remove' : true
42288         
42289         
42290     });
42291     
42292     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42293     
42294     this.items = new Roo.util.MixedCollection(false);
42295     
42296     // construct the child combo...
42297     
42298     
42299     
42300     
42301    
42302     
42303 }
42304
42305  
42306 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42307
42308     /**
42309      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42310      */
42311     
42312     lastData : false,
42313     
42314     // behavies liek a hiddne field
42315     inputType:      'hidden',
42316     /**
42317      * @cfg {Number} width The width of the box that displays the selected element
42318      */ 
42319     width:          300,
42320
42321     
42322     
42323     /**
42324      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42325      */
42326     name : false,
42327     /**
42328      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42329      */
42330     hiddenName : false,
42331       /**
42332      * @cfg {String} seperator    The value seperator normally ',' 
42333      */
42334     seperator : ',',
42335     
42336     // private the array of items that are displayed..
42337     items  : false,
42338     // private - the hidden field el.
42339     hiddenEl : false,
42340     // private - the filed el..
42341     el : false,
42342     
42343     //validateValue : function() { return true; }, // all values are ok!
42344     //onAddClick: function() { },
42345     
42346     onRender : function(ct, position) 
42347     {
42348         
42349         // create the standard hidden element
42350         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42351         
42352         
42353         // give fake names to child combo;
42354         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42355         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42356         
42357         this.combo = Roo.factory(this.combo, Roo.form);
42358         this.combo.onRender(ct, position);
42359         if (typeof(this.combo.width) != 'undefined') {
42360             this.combo.onResize(this.combo.width,0);
42361         }
42362         
42363         this.combo.initEvents();
42364         
42365         // assigned so form know we need to do this..
42366         this.store          = this.combo.store;
42367         this.valueField     = this.combo.valueField;
42368         this.displayField   = this.combo.displayField ;
42369         
42370         
42371         this.combo.wrap.addClass('x-cbarray-grp');
42372         
42373         var cbwrap = this.combo.wrap.createChild(
42374             {tag: 'div', cls: 'x-cbarray-cb'},
42375             this.combo.el.dom
42376         );
42377         
42378              
42379         this.hiddenEl = this.combo.wrap.createChild({
42380             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42381         });
42382         this.el = this.combo.wrap.createChild({
42383             tag: 'input',  type:'hidden' , name: this.name, value : ''
42384         });
42385          //   this.el.dom.removeAttribute("name");
42386         
42387         
42388         this.outerWrap = this.combo.wrap;
42389         this.wrap = cbwrap;
42390         
42391         this.outerWrap.setWidth(this.width);
42392         this.outerWrap.dom.removeChild(this.el.dom);
42393         
42394         this.wrap.dom.appendChild(this.el.dom);
42395         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42396         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42397         
42398         this.combo.trigger.setStyle('position','relative');
42399         this.combo.trigger.setStyle('left', '0px');
42400         this.combo.trigger.setStyle('top', '2px');
42401         
42402         this.combo.el.setStyle('vertical-align', 'text-bottom');
42403         
42404         //this.trigger.setStyle('vertical-align', 'top');
42405         
42406         // this should use the code from combo really... on('add' ....)
42407         if (this.adder) {
42408             
42409         
42410             this.adder = this.outerWrap.createChild(
42411                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42412             var _t = this;
42413             this.adder.on('click', function(e) {
42414                 _t.fireEvent('adderclick', this, e);
42415             }, _t);
42416         }
42417         //var _t = this;
42418         //this.adder.on('click', this.onAddClick, _t);
42419         
42420         
42421         this.combo.on('select', function(cb, rec, ix) {
42422             this.addItem(rec.data);
42423             
42424             cb.setValue('');
42425             cb.el.dom.value = '';
42426             //cb.lastData = rec.data;
42427             // add to list
42428             
42429         }, this);
42430         
42431         
42432     },
42433     
42434     
42435     getName: function()
42436     {
42437         // returns hidden if it's set..
42438         if (!this.rendered) {return ''};
42439         return  this.hiddenName ? this.hiddenName : this.name;
42440         
42441     },
42442     
42443     
42444     onResize: function(w, h){
42445         
42446         return;
42447         // not sure if this is needed..
42448         //this.combo.onResize(w,h);
42449         
42450         if(typeof w != 'number'){
42451             // we do not handle it!?!?
42452             return;
42453         }
42454         var tw = this.combo.trigger.getWidth();
42455         tw += this.addicon ? this.addicon.getWidth() : 0;
42456         tw += this.editicon ? this.editicon.getWidth() : 0;
42457         var x = w - tw;
42458         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42459             
42460         this.combo.trigger.setStyle('left', '0px');
42461         
42462         if(this.list && this.listWidth === undefined){
42463             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42464             this.list.setWidth(lw);
42465             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42466         }
42467         
42468     
42469         
42470     },
42471     
42472     addItem: function(rec)
42473     {
42474         var valueField = this.combo.valueField;
42475         var displayField = this.combo.displayField;
42476         
42477         if (this.items.indexOfKey(rec[valueField]) > -1) {
42478             //console.log("GOT " + rec.data.id);
42479             return;
42480         }
42481         
42482         var x = new Roo.form.ComboBoxArray.Item({
42483             //id : rec[this.idField],
42484             data : rec,
42485             displayField : displayField ,
42486             tipField : displayField ,
42487             cb : this
42488         });
42489         // use the 
42490         this.items.add(rec[valueField],x);
42491         // add it before the element..
42492         this.updateHiddenEl();
42493         x.render(this.outerWrap, this.wrap.dom);
42494         // add the image handler..
42495     },
42496     
42497     updateHiddenEl : function()
42498     {
42499         this.validate();
42500         if (!this.hiddenEl) {
42501             return;
42502         }
42503         var ar = [];
42504         var idField = this.combo.valueField;
42505         
42506         this.items.each(function(f) {
42507             ar.push(f.data[idField]);
42508         });
42509         this.hiddenEl.dom.value = ar.join(this.seperator);
42510         this.validate();
42511     },
42512     
42513     reset : function()
42514     {
42515         this.items.clear();
42516         
42517         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42518            el.remove();
42519         });
42520         
42521         this.el.dom.value = '';
42522         if (this.hiddenEl) {
42523             this.hiddenEl.dom.value = '';
42524         }
42525         
42526     },
42527     getValue: function()
42528     {
42529         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42530     },
42531     setValue: function(v) // not a valid action - must use addItems..
42532     {
42533         
42534         this.reset();
42535          
42536         if (this.store.isLocal && (typeof(v) == 'string')) {
42537             // then we can use the store to find the values..
42538             // comma seperated at present.. this needs to allow JSON based encoding..
42539             this.hiddenEl.value  = v;
42540             var v_ar = [];
42541             Roo.each(v.split(this.seperator), function(k) {
42542                 Roo.log("CHECK " + this.valueField + ',' + k);
42543                 var li = this.store.query(this.valueField, k);
42544                 if (!li.length) {
42545                     return;
42546                 }
42547                 var add = {};
42548                 add[this.valueField] = k;
42549                 add[this.displayField] = li.item(0).data[this.displayField];
42550                 
42551                 this.addItem(add);
42552             }, this) 
42553              
42554         }
42555         if (typeof(v) == 'object' ) {
42556             // then let's assume it's an array of objects..
42557             Roo.each(v, function(l) {
42558                 var add = l;
42559                 if (typeof(l) == 'string') {
42560                     add = {};
42561                     add[this.valueField] = l;
42562                     add[this.displayField] = l
42563                 }
42564                 this.addItem(add);
42565             }, this);
42566              
42567         }
42568         
42569         
42570     },
42571     setFromData: function(v)
42572     {
42573         // this recieves an object, if setValues is called.
42574         this.reset();
42575         this.el.dom.value = v[this.displayField];
42576         this.hiddenEl.dom.value = v[this.valueField];
42577         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42578             return;
42579         }
42580         var kv = v[this.valueField];
42581         var dv = v[this.displayField];
42582         kv = typeof(kv) != 'string' ? '' : kv;
42583         dv = typeof(dv) != 'string' ? '' : dv;
42584         
42585         
42586         var keys = kv.split(this.seperator);
42587         var display = dv.split(this.seperator);
42588         for (var i = 0 ; i < keys.length; i++) {
42589             add = {};
42590             add[this.valueField] = keys[i];
42591             add[this.displayField] = display[i];
42592             this.addItem(add);
42593         }
42594       
42595         
42596     },
42597     
42598     /**
42599      * Validates the combox array value
42600      * @return {Boolean} True if the value is valid, else false
42601      */
42602     validate : function(){
42603         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42604             this.clearInvalid();
42605             return true;
42606         }
42607         return false;
42608     },
42609     
42610     validateValue : function(value){
42611         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42612         
42613     },
42614     
42615     /*@
42616      * overide
42617      * 
42618      */
42619     isDirty : function() {
42620         if(this.disabled) {
42621             return false;
42622         }
42623         
42624         try {
42625             var d = Roo.decode(String(this.originalValue));
42626         } catch (e) {
42627             return String(this.getValue()) !== String(this.originalValue);
42628         }
42629         
42630         var originalValue = [];
42631         
42632         for (var i = 0; i < d.length; i++){
42633             originalValue.push(d[i][this.valueField]);
42634         }
42635         
42636         return String(this.getValue()) !== String(originalValue.join(this.seperator));
42637         
42638     }
42639     
42640 });
42641
42642
42643
42644 /**
42645  * @class Roo.form.ComboBoxArray.Item
42646  * @extends Roo.BoxComponent
42647  * A selected item in the list
42648  *  Fred [x]  Brian [x]  [Pick another |v]
42649  * 
42650  * @constructor
42651  * Create a new item.
42652  * @param {Object} config Configuration options
42653  */
42654  
42655 Roo.form.ComboBoxArray.Item = function(config) {
42656     config.id = Roo.id();
42657     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42658 }
42659
42660 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42661     data : {},
42662     cb: false,
42663     displayField : false,
42664     tipField : false,
42665     
42666     
42667     defaultAutoCreate : {
42668         tag: 'div',
42669         cls: 'x-cbarray-item',
42670         cn : [ 
42671             { tag: 'div' },
42672             {
42673                 tag: 'img',
42674                 width:16,
42675                 height : 16,
42676                 src : Roo.BLANK_IMAGE_URL ,
42677                 align: 'center'
42678             }
42679         ]
42680         
42681     },
42682     
42683  
42684     onRender : function(ct, position)
42685     {
42686         Roo.form.Field.superclass.onRender.call(this, ct, position);
42687         
42688         if(!this.el){
42689             var cfg = this.getAutoCreate();
42690             this.el = ct.createChild(cfg, position);
42691         }
42692         
42693         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42694         
42695         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42696             this.cb.renderer(this.data) :
42697             String.format('{0}',this.data[this.displayField]);
42698         
42699             
42700         this.el.child('div').dom.setAttribute('qtip',
42701                         String.format('{0}',this.data[this.tipField])
42702         );
42703         
42704         this.el.child('img').on('click', this.remove, this);
42705         
42706     },
42707    
42708     remove : function()
42709     {
42710         if(this.cb.disabled){
42711             return;
42712         }
42713         
42714         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42715             this.cb.items.remove(this);
42716             this.el.child('img').un('click', this.remove, this);
42717             this.el.remove();
42718             this.cb.updateHiddenEl();
42719
42720             this.cb.fireEvent('remove', this.cb, this);
42721         }
42722         
42723     }
42724 });/*
42725  * RooJS Library 1.1.1
42726  * Copyright(c) 2008-2011  Alan Knowles
42727  *
42728  * License - LGPL
42729  */
42730  
42731
42732 /**
42733  * @class Roo.form.ComboNested
42734  * @extends Roo.form.ComboBox
42735  * A combobox for that allows selection of nested items in a list,
42736  * eg.
42737  *
42738  *  Book
42739  *    -> red
42740  *    -> green
42741  *  Table
42742  *    -> square
42743  *      ->red
42744  *      ->green
42745  *    -> rectangle
42746  *      ->green
42747  *      
42748  * 
42749  * @constructor
42750  * Create a new ComboNested
42751  * @param {Object} config Configuration options
42752  */
42753 Roo.form.ComboNested = function(config){
42754     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42755     // should verify some data...
42756     // like
42757     // hiddenName = required..
42758     // displayField = required
42759     // valudField == required
42760     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42761     var _t = this;
42762     Roo.each(req, function(e) {
42763         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42764             throw "Roo.form.ComboNested : missing value for: " + e;
42765         }
42766     });
42767      
42768     
42769 };
42770
42771 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42772    
42773     /*
42774      * @config {Number} max Number of columns to show
42775      */
42776     
42777     maxColumns : 3,
42778    
42779     list : null, // the outermost div..
42780     innerLists : null, // the
42781     views : null,
42782     stores : null,
42783     // private
42784     loadingChildren : false,
42785     
42786     onRender : function(ct, position)
42787     {
42788         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42789         
42790         if(this.hiddenName){
42791             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42792                     'before', true);
42793             this.hiddenField.value =
42794                 this.hiddenValue !== undefined ? this.hiddenValue :
42795                 this.value !== undefined ? this.value : '';
42796
42797             // prevent input submission
42798             this.el.dom.removeAttribute('name');
42799              
42800              
42801         }
42802         
42803         if(Roo.isGecko){
42804             this.el.dom.setAttribute('autocomplete', 'off');
42805         }
42806
42807         var cls = 'x-combo-list';
42808
42809         this.list = new Roo.Layer({
42810             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42811         });
42812
42813         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42814         this.list.setWidth(lw);
42815         this.list.swallowEvent('mousewheel');
42816         this.assetHeight = 0;
42817
42818         if(this.title){
42819             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42820             this.assetHeight += this.header.getHeight();
42821         }
42822         this.innerLists = [];
42823         this.views = [];
42824         this.stores = [];
42825         for (var i =0 ; i < this.maxColumns; i++) {
42826             this.onRenderList( cls, i);
42827         }
42828         
42829         // always needs footer, as we are going to have an 'OK' button.
42830         this.footer = this.list.createChild({cls:cls+'-ft'});
42831         this.pageTb = new Roo.Toolbar(this.footer);  
42832         var _this = this;
42833         this.pageTb.add(  {
42834             
42835             text: 'Done',
42836             handler: function()
42837             {
42838                 _this.collapse();
42839             }
42840         });
42841         
42842         if ( this.allowBlank && !this.disableClear) {
42843             
42844             this.pageTb.add(new Roo.Toolbar.Fill(), {
42845                 cls: 'x-btn-icon x-btn-clear',
42846                 text: '&#160;',
42847                 handler: function()
42848                 {
42849                     _this.collapse();
42850                     _this.clearValue();
42851                     _this.onSelect(false, -1);
42852                 }
42853             });
42854         }
42855         if (this.footer) {
42856             this.assetHeight += this.footer.getHeight();
42857         }
42858         
42859     },
42860     onRenderList : function (  cls, i)
42861     {
42862         
42863         var lw = Math.floor(
42864                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
42865         );
42866         
42867         this.list.setWidth(lw); // default to '1'
42868
42869         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42870         //il.on('mouseover', this.onViewOver, this, { list:  i });
42871         //il.on('mousemove', this.onViewMove, this, { list:  i });
42872         il.setWidth(lw);
42873         il.setStyle({ 'overflow-x' : 'hidden'});
42874
42875         if(!this.tpl){
42876             this.tpl = new Roo.Template({
42877                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42878                 isEmpty: function (value, allValues) {
42879                     //Roo.log(value);
42880                     var dl = typeof(value.data) != 'undefined' ? value.data.length : value.length; ///json is a nested response..
42881                     return dl ? 'has-children' : 'no-children'
42882                 }
42883             });
42884         }
42885         
42886         var store  = this.store;
42887         if (i > 0) {
42888             store  = new Roo.data.SimpleStore({
42889                 //fields : this.store.reader.meta.fields,
42890                 reader : this.store.reader,
42891                 data : [ ]
42892             });
42893         }
42894         this.stores[i]  = store;
42895                   
42896         var view = this.views[i] = new Roo.View(
42897             il,
42898             this.tpl,
42899             {
42900                 singleSelect:true,
42901                 store: store,
42902                 selectedClass: this.selectedClass
42903             }
42904         );
42905         view.getEl().setWidth(lw);
42906         view.getEl().setStyle({
42907             position: i < 1 ? 'relative' : 'absolute',
42908             top: 0,
42909             left: (i * lw ) + 'px',
42910             display : i > 0 ? 'none' : 'block'
42911         });
42912         view.on('selectionchange', this.onSelectChange.createDelegate(this, {list : i }, true));
42913         view.on('dblclick', this.onDoubleClick.createDelegate(this, {list : i }, true));
42914         //view.on('click', this.onViewClick, this, { list : i });
42915
42916         store.on('beforeload', this.onBeforeLoad, this);
42917         store.on('load',  this.onLoad, this, { list  : i});
42918         store.on('loadexception', this.onLoadException, this);
42919
42920         // hide the other vies..
42921         
42922         
42923         
42924     },
42925       
42926     restrictHeight : function()
42927     {
42928         var mh = 0;
42929         Roo.each(this.innerLists, function(il,i) {
42930             var el = this.views[i].getEl();
42931             el.dom.style.height = '';
42932             var inner = el.dom;
42933             var h = Math.max(il.clientHeight, il.offsetHeight, il.scrollHeight);
42934             // only adjust heights on other ones..
42935             mh = Math.max(h, mh);
42936             if (i < 1) {
42937                 
42938                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42939                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42940                
42941             }
42942             
42943             
42944         }, this);
42945         
42946         this.list.beginUpdate();
42947         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42948         this.list.alignTo(this.el, this.listAlign);
42949         this.list.endUpdate();
42950         
42951     },
42952      
42953     
42954     // -- store handlers..
42955     // private
42956     onBeforeLoad : function()
42957     {
42958         if(!this.hasFocus){
42959             return;
42960         }
42961         this.innerLists[0].update(this.loadingText ?
42962                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
42963         this.restrictHeight();
42964         this.selectedIndex = -1;
42965     },
42966     // private
42967     onLoad : function(a,b,c,d)
42968     {
42969         if (!this.loadingChildren) {
42970             // then we are loading the top level. - hide the children
42971             for (var i = 1;i < this.views.length; i++) {
42972                 this.views[i].getEl().setStyle({ display : 'none' });
42973             }
42974             var lw = Math.floor(
42975                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
42976             );
42977         
42978              this.list.setWidth(lw); // default to '1'
42979
42980             
42981         }
42982         if(!this.hasFocus){
42983             return;
42984         }
42985         
42986         if(this.store.getCount() > 0) {
42987             this.expand();
42988             this.restrictHeight();   
42989         } else {
42990             this.onEmptyResults();
42991         }
42992         
42993         if (!this.loadingChildren) {
42994             this.selectActive();
42995         }
42996         /*
42997         this.stores[1].loadData([]);
42998         this.stores[2].loadData([]);
42999         this.views
43000         */    
43001     
43002         //this.el.focus();
43003     },
43004     
43005     
43006     // private
43007     onLoadException : function()
43008     {
43009         this.collapse();
43010         Roo.log(this.store.reader.jsonData);
43011         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
43012             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
43013         }
43014         
43015         
43016     },
43017     // no cleaning of leading spaces on blur here.
43018     cleanLeadingSpace : function(e) { },
43019     
43020
43021     onSelectChange : function (view, sels, opts )
43022     {
43023         var ix = view.getSelectedIndexes();
43024          
43025         if (opts.list > this.maxColumns - 2) {
43026             if (view.store.getCount()<  1) {
43027                 this.views[opts.list ].getEl().setStyle({ display :   'none' });
43028
43029             } else  {
43030                 if (ix.length) {
43031                     // used to clear ?? but if we are loading unselected 
43032                     this.setFromData(view.store.getAt(ix[0]).data);
43033                 }
43034                 
43035             }
43036             
43037             return;
43038         }
43039         
43040         if (!ix.length) {
43041             // this get's fired when trigger opens..
43042            // this.setFromData({});
43043             var str = this.stores[opts.list+1];
43044             str.data.clear(); // removeall wihtout the fire events..
43045             return;
43046         }
43047         
43048         var rec = view.store.getAt(ix[0]);
43049          
43050         this.setFromData(rec.data);
43051         this.fireEvent('select', this, rec, ix[0]);
43052         
43053         var lw = Math.floor(
43054              (
43055                 (this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')
43056              ) / this.maxColumns
43057         );
43058         this.loadingChildren = true;
43059         this.stores[opts.list+1].loadDataFromChildren( rec );
43060         this.loadingChildren = false;
43061         var dl = this.stores[opts.list+1]. getTotalCount();
43062         
43063         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
43064         
43065         this.views[opts.list+1].getEl().setStyle({ display : dl ? 'block' : 'none' });
43066         for (var i = opts.list+2; i < this.views.length;i++) {
43067             this.views[i].getEl().setStyle({ display : 'none' });
43068         }
43069         
43070         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
43071         this.list.setWidth(lw * (opts.list + (dl ? 2 : 1)));
43072         
43073         if (this.isLoading) {
43074            // this.selectActive(opts.list);
43075         }
43076          
43077     },
43078     
43079     
43080     
43081     
43082     onDoubleClick : function()
43083     {
43084         this.collapse(); //??
43085     },
43086     
43087      
43088     
43089     
43090     
43091     // private
43092     recordToStack : function(store, prop, value, stack)
43093     {
43094         var cstore = new Roo.data.SimpleStore({
43095             //fields : this.store.reader.meta.fields, // we need array reader.. for
43096             reader : this.store.reader,
43097             data : [ ]
43098         });
43099         var _this = this;
43100         var record  = false;
43101         var srec = false;
43102         if(store.getCount() < 1){
43103             return false;
43104         }
43105         store.each(function(r){
43106             if(r.data[prop] == value){
43107                 record = r;
43108             srec = r;
43109                 return false;
43110             }
43111             if (r.data.cn && r.data.cn.length) {
43112                 cstore.loadDataFromChildren( r);
43113                 var cret = _this.recordToStack(cstore, prop, value, stack);
43114                 if (cret !== false) {
43115                     record = cret;
43116                     srec = r;
43117                     return false;
43118                 }
43119             }
43120              
43121             return true;
43122         });
43123         if (record == false) {
43124             return false
43125         }
43126         stack.unshift(srec);
43127         return record;
43128     },
43129     
43130     /*
43131      * find the stack of stores that match our value.
43132      *
43133      * 
43134      */
43135     
43136     selectActive : function ()
43137     {
43138         // if store is not loaded, then we will need to wait for that to happen first.
43139         var stack = [];
43140         this.recordToStack(this.store, this.valueField, this.getValue(), stack);
43141         for (var i = 0; i < stack.length; i++ ) {
43142             this.views[i].select(stack[i].store.indexOf(stack[i]), false, false );
43143         }
43144         
43145     }
43146         
43147          
43148     
43149     
43150     
43151     
43152 });/*
43153  * Based on:
43154  * Ext JS Library 1.1.1
43155  * Copyright(c) 2006-2007, Ext JS, LLC.
43156  *
43157  * Originally Released Under LGPL - original licence link has changed is not relivant.
43158  *
43159  * Fork - LGPL
43160  * <script type="text/javascript">
43161  */
43162 /**
43163  * @class Roo.form.Checkbox
43164  * @extends Roo.form.Field
43165  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43166  * @constructor
43167  * Creates a new Checkbox
43168  * @param {Object} config Configuration options
43169  */
43170 Roo.form.Checkbox = function(config){
43171     Roo.form.Checkbox.superclass.constructor.call(this, config);
43172     this.addEvents({
43173         /**
43174          * @event check
43175          * Fires when the checkbox is checked or unchecked.
43176              * @param {Roo.form.Checkbox} this This checkbox
43177              * @param {Boolean} checked The new checked value
43178              */
43179         check : true
43180     });
43181 };
43182
43183 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43184     /**
43185      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43186      */
43187     focusClass : undefined,
43188     /**
43189      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43190      */
43191     fieldClass: "x-form-field",
43192     /**
43193      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43194      */
43195     checked: false,
43196     /**
43197      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43198      * {tag: "input", type: "checkbox", autocomplete: "off"})
43199      */
43200     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43201     /**
43202      * @cfg {String} boxLabel The text that appears beside the checkbox
43203      */
43204     boxLabel : "",
43205     /**
43206      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43207      */  
43208     inputValue : '1',
43209     /**
43210      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43211      */
43212      valueOff: '0', // value when not checked..
43213
43214     actionMode : 'viewEl', 
43215     //
43216     // private
43217     itemCls : 'x-menu-check-item x-form-item',
43218     groupClass : 'x-menu-group-item',
43219     inputType : 'hidden',
43220     
43221     
43222     inSetChecked: false, // check that we are not calling self...
43223     
43224     inputElement: false, // real input element?
43225     basedOn: false, // ????
43226     
43227     isFormField: true, // not sure where this is needed!!!!
43228
43229     onResize : function(){
43230         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43231         if(!this.boxLabel){
43232             this.el.alignTo(this.wrap, 'c-c');
43233         }
43234     },
43235
43236     initEvents : function(){
43237         Roo.form.Checkbox.superclass.initEvents.call(this);
43238         this.el.on("click", this.onClick,  this);
43239         this.el.on("change", this.onClick,  this);
43240     },
43241
43242
43243     getResizeEl : function(){
43244         return this.wrap;
43245     },
43246
43247     getPositionEl : function(){
43248         return this.wrap;
43249     },
43250
43251     // private
43252     onRender : function(ct, position){
43253         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43254         /*
43255         if(this.inputValue !== undefined){
43256             this.el.dom.value = this.inputValue;
43257         }
43258         */
43259         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43260         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43261         var viewEl = this.wrap.createChild({ 
43262             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43263         this.viewEl = viewEl;   
43264         this.wrap.on('click', this.onClick,  this); 
43265         
43266         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43267         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43268         
43269         
43270         
43271         if(this.boxLabel){
43272             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43273         //    viewEl.on('click', this.onClick,  this); 
43274         }
43275         //if(this.checked){
43276             this.setChecked(this.checked);
43277         //}else{
43278             //this.checked = this.el.dom;
43279         //}
43280
43281     },
43282
43283     // private
43284     initValue : Roo.emptyFn,
43285
43286     /**
43287      * Returns the checked state of the checkbox.
43288      * @return {Boolean} True if checked, else false
43289      */
43290     getValue : function(){
43291         if(this.el){
43292             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43293         }
43294         return this.valueOff;
43295         
43296     },
43297
43298         // private
43299     onClick : function(){ 
43300         if (this.disabled) {
43301             return;
43302         }
43303         this.setChecked(!this.checked);
43304
43305         //if(this.el.dom.checked != this.checked){
43306         //    this.setValue(this.el.dom.checked);
43307        // }
43308     },
43309
43310     /**
43311      * Sets the checked state of the checkbox.
43312      * On is always based on a string comparison between inputValue and the param.
43313      * @param {Boolean/String} value - the value to set 
43314      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43315      */
43316     setValue : function(v,suppressEvent){
43317         
43318         
43319         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43320         //if(this.el && this.el.dom){
43321         //    this.el.dom.checked = this.checked;
43322         //    this.el.dom.defaultChecked = this.checked;
43323         //}
43324         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43325         //this.fireEvent("check", this, this.checked);
43326     },
43327     // private..
43328     setChecked : function(state,suppressEvent)
43329     {
43330         if (this.inSetChecked) {
43331             this.checked = state;
43332             return;
43333         }
43334         
43335     
43336         if(this.wrap){
43337             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43338         }
43339         this.checked = state;
43340         if(suppressEvent !== true){
43341             this.fireEvent('check', this, state);
43342         }
43343         this.inSetChecked = true;
43344         this.el.dom.value = state ? this.inputValue : this.valueOff;
43345         this.inSetChecked = false;
43346         
43347     },
43348     // handle setting of hidden value by some other method!!?!?
43349     setFromHidden: function()
43350     {
43351         if(!this.el){
43352             return;
43353         }
43354         //console.log("SET FROM HIDDEN");
43355         //alert('setFrom hidden');
43356         this.setValue(this.el.dom.value);
43357     },
43358     
43359     onDestroy : function()
43360     {
43361         if(this.viewEl){
43362             Roo.get(this.viewEl).remove();
43363         }
43364          
43365         Roo.form.Checkbox.superclass.onDestroy.call(this);
43366     },
43367     
43368     setBoxLabel : function(str)
43369     {
43370         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43371     }
43372
43373 });/*
43374  * Based on:
43375  * Ext JS Library 1.1.1
43376  * Copyright(c) 2006-2007, Ext JS, LLC.
43377  *
43378  * Originally Released Under LGPL - original licence link has changed is not relivant.
43379  *
43380  * Fork - LGPL
43381  * <script type="text/javascript">
43382  */
43383  
43384 /**
43385  * @class Roo.form.Radio
43386  * @extends Roo.form.Checkbox
43387  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43388  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43389  * @constructor
43390  * Creates a new Radio
43391  * @param {Object} config Configuration options
43392  */
43393 Roo.form.Radio = function(){
43394     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43395 };
43396 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43397     inputType: 'radio',
43398
43399     /**
43400      * If this radio is part of a group, it will return the selected value
43401      * @return {String}
43402      */
43403     getGroupValue : function(){
43404         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43405     },
43406     
43407     
43408     onRender : function(ct, position){
43409         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43410         
43411         if(this.inputValue !== undefined){
43412             this.el.dom.value = this.inputValue;
43413         }
43414          
43415         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43416         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43417         //var viewEl = this.wrap.createChild({ 
43418         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43419         //this.viewEl = viewEl;   
43420         //this.wrap.on('click', this.onClick,  this); 
43421         
43422         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43423         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43424         
43425         
43426         
43427         if(this.boxLabel){
43428             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43429         //    viewEl.on('click', this.onClick,  this); 
43430         }
43431          if(this.checked){
43432             this.el.dom.checked =   'checked' ;
43433         }
43434          
43435     } 
43436     
43437     
43438 });//<script type="text/javascript">
43439
43440 /*
43441  * Based  Ext JS Library 1.1.1
43442  * Copyright(c) 2006-2007, Ext JS, LLC.
43443  * LGPL
43444  *
43445  */
43446  
43447 /**
43448  * @class Roo.HtmlEditorCore
43449  * @extends Roo.Component
43450  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43451  *
43452  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43453  */
43454
43455 Roo.HtmlEditorCore = function(config){
43456     
43457     
43458     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43459     
43460     
43461     this.addEvents({
43462         /**
43463          * @event initialize
43464          * Fires when the editor is fully initialized (including the iframe)
43465          * @param {Roo.HtmlEditorCore} this
43466          */
43467         initialize: true,
43468         /**
43469          * @event activate
43470          * Fires when the editor is first receives the focus. Any insertion must wait
43471          * until after this event.
43472          * @param {Roo.HtmlEditorCore} this
43473          */
43474         activate: true,
43475          /**
43476          * @event beforesync
43477          * Fires before the textarea is updated with content from the editor iframe. Return false
43478          * to cancel the sync.
43479          * @param {Roo.HtmlEditorCore} this
43480          * @param {String} html
43481          */
43482         beforesync: true,
43483          /**
43484          * @event beforepush
43485          * Fires before the iframe editor is updated with content from the textarea. Return false
43486          * to cancel the push.
43487          * @param {Roo.HtmlEditorCore} this
43488          * @param {String} html
43489          */
43490         beforepush: true,
43491          /**
43492          * @event sync
43493          * Fires when the textarea is updated with content from the editor iframe.
43494          * @param {Roo.HtmlEditorCore} this
43495          * @param {String} html
43496          */
43497         sync: true,
43498          /**
43499          * @event push
43500          * Fires when the iframe editor is updated with content from the textarea.
43501          * @param {Roo.HtmlEditorCore} this
43502          * @param {String} html
43503          */
43504         push: true,
43505         
43506         /**
43507          * @event editorevent
43508          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43509          * @param {Roo.HtmlEditorCore} this
43510          */
43511         editorevent: true
43512         
43513     });
43514     
43515     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43516     
43517     // defaults : white / black...
43518     this.applyBlacklists();
43519     
43520     
43521     
43522 };
43523
43524
43525 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43526
43527
43528      /**
43529      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43530      */
43531     
43532     owner : false,
43533     
43534      /**
43535      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43536      *                        Roo.resizable.
43537      */
43538     resizable : false,
43539      /**
43540      * @cfg {Number} height (in pixels)
43541      */   
43542     height: 300,
43543    /**
43544      * @cfg {Number} width (in pixels)
43545      */   
43546     width: 500,
43547     
43548     /**
43549      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43550      * 
43551      */
43552     stylesheets: false,
43553     
43554     // id of frame..
43555     frameId: false,
43556     
43557     // private properties
43558     validationEvent : false,
43559     deferHeight: true,
43560     initialized : false,
43561     activated : false,
43562     sourceEditMode : false,
43563     onFocus : Roo.emptyFn,
43564     iframePad:3,
43565     hideMode:'offsets',
43566     
43567     clearUp: true,
43568     
43569     // blacklist + whitelisted elements..
43570     black: false,
43571     white: false,
43572      
43573     bodyCls : '',
43574
43575     /**
43576      * Protected method that will not generally be called directly. It
43577      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43578      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43579      */
43580     getDocMarkup : function(){
43581         // body styles..
43582         var st = '';
43583         
43584         // inherit styels from page...?? 
43585         if (this.stylesheets === false) {
43586             
43587             Roo.get(document.head).select('style').each(function(node) {
43588                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43589             });
43590             
43591             Roo.get(document.head).select('link').each(function(node) { 
43592                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43593             });
43594             
43595         } else if (!this.stylesheets.length) {
43596                 // simple..
43597                 st = '<style type="text/css">' +
43598                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43599                    '</style>';
43600         } else { 
43601             st = '<style type="text/css">' +
43602                     this.stylesheets +
43603                 '</style>';
43604         }
43605         
43606         st +=  '<style type="text/css">' +
43607             'IMG { cursor: pointer } ' +
43608         '</style>';
43609
43610         var cls = 'roo-htmleditor-body';
43611         
43612         if(this.bodyCls.length){
43613             cls += ' ' + this.bodyCls;
43614         }
43615         
43616         return '<html><head>' + st  +
43617             //<style type="text/css">' +
43618             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43619             //'</style>' +
43620             ' </head><body contenteditable="true" data-enable-grammerly="true" class="' +  cls + '"></body></html>';
43621     },
43622
43623     // private
43624     onRender : function(ct, position)
43625     {
43626         var _t = this;
43627         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43628         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43629         
43630         
43631         this.el.dom.style.border = '0 none';
43632         this.el.dom.setAttribute('tabIndex', -1);
43633         this.el.addClass('x-hidden hide');
43634         
43635         
43636         
43637         if(Roo.isIE){ // fix IE 1px bogus margin
43638             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43639         }
43640        
43641         
43642         this.frameId = Roo.id();
43643         
43644          
43645         
43646         var iframe = this.owner.wrap.createChild({
43647             tag: 'iframe',
43648             cls: 'form-control', // bootstrap..
43649             id: this.frameId,
43650             name: this.frameId,
43651             frameBorder : 'no',
43652             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43653         }, this.el
43654         );
43655         
43656         
43657         this.iframe = iframe.dom;
43658
43659          this.assignDocWin();
43660         
43661         this.doc.designMode = 'on';
43662        
43663         this.doc.open();
43664         this.doc.write(this.getDocMarkup());
43665         this.doc.close();
43666
43667         
43668         var task = { // must defer to wait for browser to be ready
43669             run : function(){
43670                 //console.log("run task?" + this.doc.readyState);
43671                 this.assignDocWin();
43672                 if(this.doc.body || this.doc.readyState == 'complete'){
43673                     try {
43674                         this.doc.designMode="on";
43675                     } catch (e) {
43676                         return;
43677                     }
43678                     Roo.TaskMgr.stop(task);
43679                     this.initEditor.defer(10, this);
43680                 }
43681             },
43682             interval : 10,
43683             duration: 10000,
43684             scope: this
43685         };
43686         Roo.TaskMgr.start(task);
43687
43688     },
43689
43690     // private
43691     onResize : function(w, h)
43692     {
43693          Roo.log('resize: ' +w + ',' + h );
43694         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43695         if(!this.iframe){
43696             return;
43697         }
43698         if(typeof w == 'number'){
43699             
43700             this.iframe.style.width = w + 'px';
43701         }
43702         if(typeof h == 'number'){
43703             
43704             this.iframe.style.height = h + 'px';
43705             if(this.doc){
43706                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43707             }
43708         }
43709         
43710     },
43711
43712     /**
43713      * Toggles the editor between standard and source edit mode.
43714      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43715      */
43716     toggleSourceEdit : function(sourceEditMode){
43717         
43718         this.sourceEditMode = sourceEditMode === true;
43719         
43720         if(this.sourceEditMode){
43721  
43722             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43723             
43724         }else{
43725             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43726             //this.iframe.className = '';
43727             this.deferFocus();
43728         }
43729         //this.setSize(this.owner.wrap.getSize());
43730         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43731     },
43732
43733     
43734   
43735
43736     /**
43737      * Protected method that will not generally be called directly. If you need/want
43738      * custom HTML cleanup, this is the method you should override.
43739      * @param {String} html The HTML to be cleaned
43740      * return {String} The cleaned HTML
43741      */
43742     cleanHtml : function(html){
43743         html = String(html);
43744         if(html.length > 5){
43745             if(Roo.isSafari){ // strip safari nonsense
43746                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43747             }
43748         }
43749         if(html == '&nbsp;'){
43750             html = '';
43751         }
43752         return html;
43753     },
43754
43755     /**
43756      * HTML Editor -> Textarea
43757      * Protected method that will not generally be called directly. Syncs the contents
43758      * of the editor iframe with the textarea.
43759      */
43760     syncValue : function(){
43761         if(this.initialized){
43762             var bd = (this.doc.body || this.doc.documentElement);
43763             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43764             var html = bd.innerHTML;
43765             if(Roo.isSafari){
43766                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43767                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43768                 if(m && m[1]){
43769                     html = '<div style="'+m[0]+'">' + html + '</div>';
43770                 }
43771             }
43772             html = this.cleanHtml(html);
43773             // fix up the special chars.. normaly like back quotes in word...
43774             // however we do not want to do this with chinese..
43775             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43776                 
43777                 var cc = match.charCodeAt();
43778
43779                 // Get the character value, handling surrogate pairs
43780                 if (match.length == 2) {
43781                     // It's a surrogate pair, calculate the Unicode code point
43782                     var high = match.charCodeAt(0) - 0xD800;
43783                     var low  = match.charCodeAt(1) - 0xDC00;
43784                     cc = (high * 0x400) + low + 0x10000;
43785                 }  else if (
43786                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43787                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43788                     (cc >= 0xf900 && cc < 0xfb00 )
43789                 ) {
43790                         return match;
43791                 }  
43792          
43793                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43794                 return "&#" + cc + ";";
43795                 
43796                 
43797             });
43798             
43799             
43800              
43801             if(this.owner.fireEvent('beforesync', this, html) !== false){
43802                 this.el.dom.value = html;
43803                 this.owner.fireEvent('sync', this, html);
43804             }
43805         }
43806     },
43807
43808     /**
43809      * Protected method that will not generally be called directly. Pushes the value of the textarea
43810      * into the iframe editor.
43811      */
43812     pushValue : function(){
43813         if(this.initialized){
43814             var v = this.el.dom.value.trim();
43815             
43816 //            if(v.length < 1){
43817 //                v = '&#160;';
43818 //            }
43819             
43820             if(this.owner.fireEvent('beforepush', this, v) !== false){
43821                 var d = (this.doc.body || this.doc.documentElement);
43822                 d.innerHTML = v;
43823                 this.cleanUpPaste();
43824                 this.el.dom.value = d.innerHTML;
43825                 this.owner.fireEvent('push', this, v);
43826             }
43827         }
43828     },
43829
43830     // private
43831     deferFocus : function(){
43832         this.focus.defer(10, this);
43833     },
43834
43835     // doc'ed in Field
43836     focus : function(){
43837         if(this.win && !this.sourceEditMode){
43838             this.win.focus();
43839         }else{
43840             this.el.focus();
43841         }
43842     },
43843     
43844     assignDocWin: function()
43845     {
43846         var iframe = this.iframe;
43847         
43848          if(Roo.isIE){
43849             this.doc = iframe.contentWindow.document;
43850             this.win = iframe.contentWindow;
43851         } else {
43852 //            if (!Roo.get(this.frameId)) {
43853 //                return;
43854 //            }
43855 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43856 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43857             
43858             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43859                 return;
43860             }
43861             
43862             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43863             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43864         }
43865     },
43866     
43867     // private
43868     initEditor : function(){
43869         //console.log("INIT EDITOR");
43870         this.assignDocWin();
43871         
43872         
43873         
43874         this.doc.designMode="on";
43875         this.doc.open();
43876         this.doc.write(this.getDocMarkup());
43877         this.doc.close();
43878         
43879         var dbody = (this.doc.body || this.doc.documentElement);
43880         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43881         // this copies styles from the containing element into thsi one..
43882         // not sure why we need all of this..
43883         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43884         
43885         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43886         //ss['background-attachment'] = 'fixed'; // w3c
43887         dbody.bgProperties = 'fixed'; // ie
43888         //Roo.DomHelper.applyStyles(dbody, ss);
43889         Roo.EventManager.on(this.doc, {
43890             //'mousedown': this.onEditorEvent,
43891             'mouseup': this.onEditorEvent,
43892             'dblclick': this.onEditorEvent,
43893             'click': this.onEditorEvent,
43894             'keyup': this.onEditorEvent,
43895             buffer:100,
43896             scope: this
43897         });
43898         if(Roo.isGecko){
43899             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43900         }
43901         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43902             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43903         }
43904         this.initialized = true;
43905
43906         this.owner.fireEvent('initialize', this);
43907         this.pushValue();
43908     },
43909
43910     // private
43911     onDestroy : function(){
43912         
43913         
43914         
43915         if(this.rendered){
43916             
43917             //for (var i =0; i < this.toolbars.length;i++) {
43918             //    // fixme - ask toolbars for heights?
43919             //    this.toolbars[i].onDestroy();
43920            // }
43921             
43922             //this.wrap.dom.innerHTML = '';
43923             //this.wrap.remove();
43924         }
43925     },
43926
43927     // private
43928     onFirstFocus : function(){
43929         
43930         this.assignDocWin();
43931         
43932         
43933         this.activated = true;
43934          
43935     
43936         if(Roo.isGecko){ // prevent silly gecko errors
43937             this.win.focus();
43938             var s = this.win.getSelection();
43939             if(!s.focusNode || s.focusNode.nodeType != 3){
43940                 var r = s.getRangeAt(0);
43941                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43942                 r.collapse(true);
43943                 this.deferFocus();
43944             }
43945             try{
43946                 this.execCmd('useCSS', true);
43947                 this.execCmd('styleWithCSS', false);
43948             }catch(e){}
43949         }
43950         this.owner.fireEvent('activate', this);
43951     },
43952
43953     // private
43954     adjustFont: function(btn){
43955         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43956         //if(Roo.isSafari){ // safari
43957         //    adjust *= 2;
43958        // }
43959         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43960         if(Roo.isSafari){ // safari
43961             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43962             v =  (v < 10) ? 10 : v;
43963             v =  (v > 48) ? 48 : v;
43964             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43965             
43966         }
43967         
43968         
43969         v = Math.max(1, v+adjust);
43970         
43971         this.execCmd('FontSize', v  );
43972     },
43973
43974     onEditorEvent : function(e)
43975     {
43976         this.owner.fireEvent('editorevent', this, e);
43977       //  this.updateToolbar();
43978         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43979     },
43980
43981     insertTag : function(tg)
43982     {
43983         // could be a bit smarter... -> wrap the current selected tRoo..
43984         if (tg.toLowerCase() == 'span' ||
43985             tg.toLowerCase() == 'code' ||
43986             tg.toLowerCase() == 'sup' ||
43987             tg.toLowerCase() == 'sub' 
43988             ) {
43989             
43990             range = this.createRange(this.getSelection());
43991             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43992             wrappingNode.appendChild(range.extractContents());
43993             range.insertNode(wrappingNode);
43994
43995             return;
43996             
43997             
43998             
43999         }
44000         this.execCmd("formatblock",   tg);
44001         
44002     },
44003     
44004     insertText : function(txt)
44005     {
44006         
44007         
44008         var range = this.createRange();
44009         range.deleteContents();
44010                //alert(Sender.getAttribute('label'));
44011                
44012         range.insertNode(this.doc.createTextNode(txt));
44013     } ,
44014     
44015      
44016
44017     /**
44018      * Executes a Midas editor command on the editor document and performs necessary focus and
44019      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
44020      * @param {String} cmd The Midas command
44021      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
44022      */
44023     relayCmd : function(cmd, value){
44024         this.win.focus();
44025         this.execCmd(cmd, value);
44026         this.owner.fireEvent('editorevent', this);
44027         //this.updateToolbar();
44028         this.owner.deferFocus();
44029     },
44030
44031     /**
44032      * Executes a Midas editor command directly on the editor document.
44033      * For visual commands, you should use {@link #relayCmd} instead.
44034      * <b>This should only be called after the editor is initialized.</b>
44035      * @param {String} cmd The Midas command
44036      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
44037      */
44038     execCmd : function(cmd, value){
44039         this.doc.execCommand(cmd, false, value === undefined ? null : value);
44040         this.syncValue();
44041     },
44042  
44043  
44044    
44045     /**
44046      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
44047      * to insert tRoo.
44048      * @param {String} text | dom node.. 
44049      */
44050     insertAtCursor : function(text)
44051     {
44052         
44053         if(!this.activated){
44054             return;
44055         }
44056         /*
44057         if(Roo.isIE){
44058             this.win.focus();
44059             var r = this.doc.selection.createRange();
44060             if(r){
44061                 r.collapse(true);
44062                 r.pasteHTML(text);
44063                 this.syncValue();
44064                 this.deferFocus();
44065             
44066             }
44067             return;
44068         }
44069         */
44070         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
44071             this.win.focus();
44072             
44073             
44074             // from jquery ui (MIT licenced)
44075             var range, node;
44076             var win = this.win;
44077             
44078             if (win.getSelection && win.getSelection().getRangeAt) {
44079                 range = win.getSelection().getRangeAt(0);
44080                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
44081                 range.insertNode(node);
44082             } else if (win.document.selection && win.document.selection.createRange) {
44083                 // no firefox support
44084                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
44085                 win.document.selection.createRange().pasteHTML(txt);
44086             } else {
44087                 // no firefox support
44088                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
44089                 this.execCmd('InsertHTML', txt);
44090             } 
44091             
44092             this.syncValue();
44093             
44094             this.deferFocus();
44095         }
44096     },
44097  // private
44098     mozKeyPress : function(e){
44099         if(e.ctrlKey){
44100             var c = e.getCharCode(), cmd;
44101           
44102             if(c > 0){
44103                 c = String.fromCharCode(c).toLowerCase();
44104                 switch(c){
44105                     case 'b':
44106                         cmd = 'bold';
44107                         break;
44108                     case 'i':
44109                         cmd = 'italic';
44110                         break;
44111                     
44112                     case 'u':
44113                         cmd = 'underline';
44114                         break;
44115                     
44116                     case 'v':
44117                         this.cleanUpPaste.defer(100, this);
44118                         return;
44119                         
44120                 }
44121                 if(cmd){
44122                     this.win.focus();
44123                     this.execCmd(cmd);
44124                     this.deferFocus();
44125                     e.preventDefault();
44126                 }
44127                 
44128             }
44129         }
44130     },
44131
44132     // private
44133     fixKeys : function(){ // load time branching for fastest keydown performance
44134         if(Roo.isIE){
44135             return function(e){
44136                 var k = e.getKey(), r;
44137                 if(k == e.TAB){
44138                     e.stopEvent();
44139                     r = this.doc.selection.createRange();
44140                     if(r){
44141                         r.collapse(true);
44142                         r.pasteHTML('&#160;&#160;&#160;&#160;');
44143                         this.deferFocus();
44144                     }
44145                     return;
44146                 }
44147                 
44148                 if(k == e.ENTER){
44149                     r = this.doc.selection.createRange();
44150                     if(r){
44151                         var target = r.parentElement();
44152                         if(!target || target.tagName.toLowerCase() != 'li'){
44153                             e.stopEvent();
44154                             r.pasteHTML('<br />');
44155                             r.collapse(false);
44156                             r.select();
44157                         }
44158                     }
44159                 }
44160                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44161                     this.cleanUpPaste.defer(100, this);
44162                     return;
44163                 }
44164                 
44165                 
44166             };
44167         }else if(Roo.isOpera){
44168             return function(e){
44169                 var k = e.getKey();
44170                 if(k == e.TAB){
44171                     e.stopEvent();
44172                     this.win.focus();
44173                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44174                     this.deferFocus();
44175                 }
44176                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44177                     this.cleanUpPaste.defer(100, this);
44178                     return;
44179                 }
44180                 
44181             };
44182         }else if(Roo.isSafari){
44183             return function(e){
44184                 var k = e.getKey();
44185                 
44186                 if(k == e.TAB){
44187                     e.stopEvent();
44188                     this.execCmd('InsertText','\t');
44189                     this.deferFocus();
44190                     return;
44191                 }
44192                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44193                     this.cleanUpPaste.defer(100, this);
44194                     return;
44195                 }
44196                 
44197              };
44198         }
44199     }(),
44200     
44201     getAllAncestors: function()
44202     {
44203         var p = this.getSelectedNode();
44204         var a = [];
44205         if (!p) {
44206             a.push(p); // push blank onto stack..
44207             p = this.getParentElement();
44208         }
44209         
44210         
44211         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44212             a.push(p);
44213             p = p.parentNode;
44214         }
44215         a.push(this.doc.body);
44216         return a;
44217     },
44218     lastSel : false,
44219     lastSelNode : false,
44220     
44221     
44222     getSelection : function() 
44223     {
44224         this.assignDocWin();
44225         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44226     },
44227     
44228     getSelectedNode: function() 
44229     {
44230         // this may only work on Gecko!!!
44231         
44232         // should we cache this!!!!
44233         
44234         
44235         
44236          
44237         var range = this.createRange(this.getSelection()).cloneRange();
44238         
44239         if (Roo.isIE) {
44240             var parent = range.parentElement();
44241             while (true) {
44242                 var testRange = range.duplicate();
44243                 testRange.moveToElementText(parent);
44244                 if (testRange.inRange(range)) {
44245                     break;
44246                 }
44247                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44248                     break;
44249                 }
44250                 parent = parent.parentElement;
44251             }
44252             return parent;
44253         }
44254         
44255         // is ancestor a text element.
44256         var ac =  range.commonAncestorContainer;
44257         if (ac.nodeType == 3) {
44258             ac = ac.parentNode;
44259         }
44260         
44261         var ar = ac.childNodes;
44262          
44263         var nodes = [];
44264         var other_nodes = [];
44265         var has_other_nodes = false;
44266         for (var i=0;i<ar.length;i++) {
44267             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44268                 continue;
44269             }
44270             // fullly contained node.
44271             
44272             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44273                 nodes.push(ar[i]);
44274                 continue;
44275             }
44276             
44277             // probably selected..
44278             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44279                 other_nodes.push(ar[i]);
44280                 continue;
44281             }
44282             // outer..
44283             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44284                 continue;
44285             }
44286             
44287             
44288             has_other_nodes = true;
44289         }
44290         if (!nodes.length && other_nodes.length) {
44291             nodes= other_nodes;
44292         }
44293         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44294             return false;
44295         }
44296         
44297         return nodes[0];
44298     },
44299     createRange: function(sel)
44300     {
44301         // this has strange effects when using with 
44302         // top toolbar - not sure if it's a great idea.
44303         //this.editor.contentWindow.focus();
44304         if (typeof sel != "undefined") {
44305             try {
44306                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44307             } catch(e) {
44308                 return this.doc.createRange();
44309             }
44310         } else {
44311             return this.doc.createRange();
44312         }
44313     },
44314     getParentElement: function()
44315     {
44316         
44317         this.assignDocWin();
44318         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44319         
44320         var range = this.createRange(sel);
44321          
44322         try {
44323             var p = range.commonAncestorContainer;
44324             while (p.nodeType == 3) { // text node
44325                 p = p.parentNode;
44326             }
44327             return p;
44328         } catch (e) {
44329             return null;
44330         }
44331     
44332     },
44333     /***
44334      *
44335      * Range intersection.. the hard stuff...
44336      *  '-1' = before
44337      *  '0' = hits..
44338      *  '1' = after.
44339      *         [ -- selected range --- ]
44340      *   [fail]                        [fail]
44341      *
44342      *    basically..
44343      *      if end is before start or  hits it. fail.
44344      *      if start is after end or hits it fail.
44345      *
44346      *   if either hits (but other is outside. - then it's not 
44347      *   
44348      *    
44349      **/
44350     
44351     
44352     // @see http://www.thismuchiknow.co.uk/?p=64.
44353     rangeIntersectsNode : function(range, node)
44354     {
44355         var nodeRange = node.ownerDocument.createRange();
44356         try {
44357             nodeRange.selectNode(node);
44358         } catch (e) {
44359             nodeRange.selectNodeContents(node);
44360         }
44361     
44362         var rangeStartRange = range.cloneRange();
44363         rangeStartRange.collapse(true);
44364     
44365         var rangeEndRange = range.cloneRange();
44366         rangeEndRange.collapse(false);
44367     
44368         var nodeStartRange = nodeRange.cloneRange();
44369         nodeStartRange.collapse(true);
44370     
44371         var nodeEndRange = nodeRange.cloneRange();
44372         nodeEndRange.collapse(false);
44373     
44374         return rangeStartRange.compareBoundaryPoints(
44375                  Range.START_TO_START, nodeEndRange) == -1 &&
44376                rangeEndRange.compareBoundaryPoints(
44377                  Range.START_TO_START, nodeStartRange) == 1;
44378         
44379          
44380     },
44381     rangeCompareNode : function(range, node)
44382     {
44383         var nodeRange = node.ownerDocument.createRange();
44384         try {
44385             nodeRange.selectNode(node);
44386         } catch (e) {
44387             nodeRange.selectNodeContents(node);
44388         }
44389         
44390         
44391         range.collapse(true);
44392     
44393         nodeRange.collapse(true);
44394      
44395         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44396         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44397          
44398         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44399         
44400         var nodeIsBefore   =  ss == 1;
44401         var nodeIsAfter    = ee == -1;
44402         
44403         if (nodeIsBefore && nodeIsAfter) {
44404             return 0; // outer
44405         }
44406         if (!nodeIsBefore && nodeIsAfter) {
44407             return 1; //right trailed.
44408         }
44409         
44410         if (nodeIsBefore && !nodeIsAfter) {
44411             return 2;  // left trailed.
44412         }
44413         // fully contined.
44414         return 3;
44415     },
44416
44417     // private? - in a new class?
44418     cleanUpPaste :  function()
44419     {
44420         // cleans up the whole document..
44421         Roo.log('cleanuppaste');
44422         
44423         this.cleanUpChildren(this.doc.body);
44424         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44425         if (clean != this.doc.body.innerHTML) {
44426             this.doc.body.innerHTML = clean;
44427         }
44428         
44429     },
44430     
44431     cleanWordChars : function(input) {// change the chars to hex code
44432         var he = Roo.HtmlEditorCore;
44433         
44434         var output = input;
44435         Roo.each(he.swapCodes, function(sw) { 
44436             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44437             
44438             output = output.replace(swapper, sw[1]);
44439         });
44440         
44441         return output;
44442     },
44443     
44444     
44445     cleanUpChildren : function (n)
44446     {
44447         if (!n.childNodes.length) {
44448             return;
44449         }
44450         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44451            this.cleanUpChild(n.childNodes[i]);
44452         }
44453     },
44454     
44455     
44456         
44457     
44458     cleanUpChild : function (node)
44459     {
44460         var ed = this;
44461         //console.log(node);
44462         if (node.nodeName == "#text") {
44463             // clean up silly Windows -- stuff?
44464             return; 
44465         }
44466         if (node.nodeName == "#comment") {
44467             node.parentNode.removeChild(node);
44468             // clean up silly Windows -- stuff?
44469             return; 
44470         }
44471         var lcname = node.tagName.toLowerCase();
44472         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44473         // whitelist of tags..
44474         
44475         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44476             // remove node.
44477             node.parentNode.removeChild(node);
44478             return;
44479             
44480         }
44481         
44482         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44483         
44484         // spans with no attributes - just remove them..
44485         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44486             remove_keep_children = true;
44487         }
44488         
44489         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44490         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44491         
44492         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44493         //    remove_keep_children = true;
44494         //}
44495         
44496         if (remove_keep_children) {
44497             this.cleanUpChildren(node);
44498             // inserts everything just before this node...
44499             while (node.childNodes.length) {
44500                 var cn = node.childNodes[0];
44501                 node.removeChild(cn);
44502                 node.parentNode.insertBefore(cn, node);
44503             }
44504             node.parentNode.removeChild(node);
44505             return;
44506         }
44507         
44508         if (!node.attributes || !node.attributes.length) {
44509             
44510           
44511             
44512             
44513             this.cleanUpChildren(node);
44514             return;
44515         }
44516         
44517         function cleanAttr(n,v)
44518         {
44519             
44520             if (v.match(/^\./) || v.match(/^\//)) {
44521                 return;
44522             }
44523             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44524                 return;
44525             }
44526             if (v.match(/^#/)) {
44527                 return;
44528             }
44529             if (v.match(/^\{/)) { // allow template editing.
44530                 return;
44531             }
44532 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44533             node.removeAttribute(n);
44534             
44535         }
44536         
44537         var cwhite = this.cwhite;
44538         var cblack = this.cblack;
44539             
44540         function cleanStyle(n,v)
44541         {
44542             if (v.match(/expression/)) { //XSS?? should we even bother..
44543                 node.removeAttribute(n);
44544                 return;
44545             }
44546             
44547             var parts = v.split(/;/);
44548             var clean = [];
44549             
44550             Roo.each(parts, function(p) {
44551                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44552                 if (!p.length) {
44553                     return true;
44554                 }
44555                 var l = p.split(':').shift().replace(/\s+/g,'');
44556                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44557                 
44558                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44559 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44560                     //node.removeAttribute(n);
44561                     return true;
44562                 }
44563                 //Roo.log()
44564                 // only allow 'c whitelisted system attributes'
44565                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44566 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44567                     //node.removeAttribute(n);
44568                     return true;
44569                 }
44570                 
44571                 
44572                  
44573                 
44574                 clean.push(p);
44575                 return true;
44576             });
44577             if (clean.length) { 
44578                 node.setAttribute(n, clean.join(';'));
44579             } else {
44580                 node.removeAttribute(n);
44581             }
44582             
44583         }
44584         
44585         
44586         for (var i = node.attributes.length-1; i > -1 ; i--) {
44587             var a = node.attributes[i];
44588             //console.log(a);
44589             
44590             if (a.name.toLowerCase().substr(0,2)=='on')  {
44591                 node.removeAttribute(a.name);
44592                 continue;
44593             }
44594             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44595                 node.removeAttribute(a.name);
44596                 continue;
44597             }
44598             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44599                 cleanAttr(a.name,a.value); // fixme..
44600                 continue;
44601             }
44602             if (a.name == 'style') {
44603                 cleanStyle(a.name,a.value);
44604                 continue;
44605             }
44606             /// clean up MS crap..
44607             // tecnically this should be a list of valid class'es..
44608             
44609             
44610             if (a.name == 'class') {
44611                 if (a.value.match(/^Mso/)) {
44612                     node.removeAttribute('class');
44613                 }
44614                 
44615                 if (a.value.match(/^body$/)) {
44616                     node.removeAttribute('class');
44617                 }
44618                 continue;
44619             }
44620             
44621             // style cleanup!?
44622             // class cleanup?
44623             
44624         }
44625         
44626         
44627         this.cleanUpChildren(node);
44628         
44629         
44630     },
44631     
44632     /**
44633      * Clean up MS wordisms...
44634      */
44635     cleanWord : function(node)
44636     {
44637         if (!node) {
44638             this.cleanWord(this.doc.body);
44639             return;
44640         }
44641         
44642         if(
44643                 node.nodeName == 'SPAN' &&
44644                 !node.hasAttributes() &&
44645                 node.childNodes.length == 1 &&
44646                 node.firstChild.nodeName == "#text"  
44647         ) {
44648             var textNode = node.firstChild;
44649             node.removeChild(textNode);
44650             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44651                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44652             }
44653             node.parentNode.insertBefore(textNode, node);
44654             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44655                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44656             }
44657             node.parentNode.removeChild(node);
44658         }
44659         
44660         if (node.nodeName == "#text") {
44661             // clean up silly Windows -- stuff?
44662             return; 
44663         }
44664         if (node.nodeName == "#comment") {
44665             node.parentNode.removeChild(node);
44666             // clean up silly Windows -- stuff?
44667             return; 
44668         }
44669         
44670         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44671             node.parentNode.removeChild(node);
44672             return;
44673         }
44674         //Roo.log(node.tagName);
44675         // remove - but keep children..
44676         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44677             //Roo.log('-- removed');
44678             while (node.childNodes.length) {
44679                 var cn = node.childNodes[0];
44680                 node.removeChild(cn);
44681                 node.parentNode.insertBefore(cn, node);
44682                 // move node to parent - and clean it..
44683                 this.cleanWord(cn);
44684             }
44685             node.parentNode.removeChild(node);
44686             /// no need to iterate chidlren = it's got none..
44687             //this.iterateChildren(node, this.cleanWord);
44688             return;
44689         }
44690         // clean styles
44691         if (node.className.length) {
44692             
44693             var cn = node.className.split(/\W+/);
44694             var cna = [];
44695             Roo.each(cn, function(cls) {
44696                 if (cls.match(/Mso[a-zA-Z]+/)) {
44697                     return;
44698                 }
44699                 cna.push(cls);
44700             });
44701             node.className = cna.length ? cna.join(' ') : '';
44702             if (!cna.length) {
44703                 node.removeAttribute("class");
44704             }
44705         }
44706         
44707         if (node.hasAttribute("lang")) {
44708             node.removeAttribute("lang");
44709         }
44710         
44711         if (node.hasAttribute("style")) {
44712             
44713             var styles = node.getAttribute("style").split(";");
44714             var nstyle = [];
44715             Roo.each(styles, function(s) {
44716                 if (!s.match(/:/)) {
44717                     return;
44718                 }
44719                 var kv = s.split(":");
44720                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44721                     return;
44722                 }
44723                 // what ever is left... we allow.
44724                 nstyle.push(s);
44725             });
44726             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44727             if (!nstyle.length) {
44728                 node.removeAttribute('style');
44729             }
44730         }
44731         this.iterateChildren(node, this.cleanWord);
44732         
44733         
44734         
44735     },
44736     /**
44737      * iterateChildren of a Node, calling fn each time, using this as the scole..
44738      * @param {DomNode} node node to iterate children of.
44739      * @param {Function} fn method of this class to call on each item.
44740      */
44741     iterateChildren : function(node, fn)
44742     {
44743         if (!node.childNodes.length) {
44744                 return;
44745         }
44746         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44747            fn.call(this, node.childNodes[i])
44748         }
44749     },
44750     
44751     
44752     /**
44753      * cleanTableWidths.
44754      *
44755      * Quite often pasting from word etc.. results in tables with column and widths.
44756      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44757      *
44758      */
44759     cleanTableWidths : function(node)
44760     {
44761          
44762          
44763         if (!node) {
44764             this.cleanTableWidths(this.doc.body);
44765             return;
44766         }
44767         
44768         // ignore list...
44769         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44770             return; 
44771         }
44772         Roo.log(node.tagName);
44773         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44774             this.iterateChildren(node, this.cleanTableWidths);
44775             return;
44776         }
44777         if (node.hasAttribute('width')) {
44778             node.removeAttribute('width');
44779         }
44780         
44781          
44782         if (node.hasAttribute("style")) {
44783             // pretty basic...
44784             
44785             var styles = node.getAttribute("style").split(";");
44786             var nstyle = [];
44787             Roo.each(styles, function(s) {
44788                 if (!s.match(/:/)) {
44789                     return;
44790                 }
44791                 var kv = s.split(":");
44792                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44793                     return;
44794                 }
44795                 // what ever is left... we allow.
44796                 nstyle.push(s);
44797             });
44798             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44799             if (!nstyle.length) {
44800                 node.removeAttribute('style');
44801             }
44802         }
44803         
44804         this.iterateChildren(node, this.cleanTableWidths);
44805         
44806         
44807     },
44808     
44809     
44810     
44811     
44812     domToHTML : function(currentElement, depth, nopadtext) {
44813         
44814         depth = depth || 0;
44815         nopadtext = nopadtext || false;
44816     
44817         if (!currentElement) {
44818             return this.domToHTML(this.doc.body);
44819         }
44820         
44821         //Roo.log(currentElement);
44822         var j;
44823         var allText = false;
44824         var nodeName = currentElement.nodeName;
44825         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44826         
44827         if  (nodeName == '#text') {
44828             
44829             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44830         }
44831         
44832         
44833         var ret = '';
44834         if (nodeName != 'BODY') {
44835              
44836             var i = 0;
44837             // Prints the node tagName, such as <A>, <IMG>, etc
44838             if (tagName) {
44839                 var attr = [];
44840                 for(i = 0; i < currentElement.attributes.length;i++) {
44841                     // quoting?
44842                     var aname = currentElement.attributes.item(i).name;
44843                     if (!currentElement.attributes.item(i).value.length) {
44844                         continue;
44845                     }
44846                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44847                 }
44848                 
44849                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44850             } 
44851             else {
44852                 
44853                 // eack
44854             }
44855         } else {
44856             tagName = false;
44857         }
44858         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44859             return ret;
44860         }
44861         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44862             nopadtext = true;
44863         }
44864         
44865         
44866         // Traverse the tree
44867         i = 0;
44868         var currentElementChild = currentElement.childNodes.item(i);
44869         var allText = true;
44870         var innerHTML  = '';
44871         lastnode = '';
44872         while (currentElementChild) {
44873             // Formatting code (indent the tree so it looks nice on the screen)
44874             var nopad = nopadtext;
44875             if (lastnode == 'SPAN') {
44876                 nopad  = true;
44877             }
44878             // text
44879             if  (currentElementChild.nodeName == '#text') {
44880                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44881                 toadd = nopadtext ? toadd : toadd.trim();
44882                 if (!nopad && toadd.length > 80) {
44883                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44884                 }
44885                 innerHTML  += toadd;
44886                 
44887                 i++;
44888                 currentElementChild = currentElement.childNodes.item(i);
44889                 lastNode = '';
44890                 continue;
44891             }
44892             allText = false;
44893             
44894             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44895                 
44896             // Recursively traverse the tree structure of the child node
44897             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44898             lastnode = currentElementChild.nodeName;
44899             i++;
44900             currentElementChild=currentElement.childNodes.item(i);
44901         }
44902         
44903         ret += innerHTML;
44904         
44905         if (!allText) {
44906                 // The remaining code is mostly for formatting the tree
44907             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44908         }
44909         
44910         
44911         if (tagName) {
44912             ret+= "</"+tagName+">";
44913         }
44914         return ret;
44915         
44916     },
44917         
44918     applyBlacklists : function()
44919     {
44920         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44921         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44922         
44923         this.white = [];
44924         this.black = [];
44925         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44926             if (b.indexOf(tag) > -1) {
44927                 return;
44928             }
44929             this.white.push(tag);
44930             
44931         }, this);
44932         
44933         Roo.each(w, function(tag) {
44934             if (b.indexOf(tag) > -1) {
44935                 return;
44936             }
44937             if (this.white.indexOf(tag) > -1) {
44938                 return;
44939             }
44940             this.white.push(tag);
44941             
44942         }, this);
44943         
44944         
44945         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44946             if (w.indexOf(tag) > -1) {
44947                 return;
44948             }
44949             this.black.push(tag);
44950             
44951         }, this);
44952         
44953         Roo.each(b, function(tag) {
44954             if (w.indexOf(tag) > -1) {
44955                 return;
44956             }
44957             if (this.black.indexOf(tag) > -1) {
44958                 return;
44959             }
44960             this.black.push(tag);
44961             
44962         }, this);
44963         
44964         
44965         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44966         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44967         
44968         this.cwhite = [];
44969         this.cblack = [];
44970         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44971             if (b.indexOf(tag) > -1) {
44972                 return;
44973             }
44974             this.cwhite.push(tag);
44975             
44976         }, this);
44977         
44978         Roo.each(w, function(tag) {
44979             if (b.indexOf(tag) > -1) {
44980                 return;
44981             }
44982             if (this.cwhite.indexOf(tag) > -1) {
44983                 return;
44984             }
44985             this.cwhite.push(tag);
44986             
44987         }, this);
44988         
44989         
44990         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44991             if (w.indexOf(tag) > -1) {
44992                 return;
44993             }
44994             this.cblack.push(tag);
44995             
44996         }, this);
44997         
44998         Roo.each(b, function(tag) {
44999             if (w.indexOf(tag) > -1) {
45000                 return;
45001             }
45002             if (this.cblack.indexOf(tag) > -1) {
45003                 return;
45004             }
45005             this.cblack.push(tag);
45006             
45007         }, this);
45008     },
45009     
45010     setStylesheets : function(stylesheets)
45011     {
45012         if(typeof(stylesheets) == 'string'){
45013             Roo.get(this.iframe.contentDocument.head).createChild({
45014                 tag : 'link',
45015                 rel : 'stylesheet',
45016                 type : 'text/css',
45017                 href : stylesheets
45018             });
45019             
45020             return;
45021         }
45022         var _this = this;
45023      
45024         Roo.each(stylesheets, function(s) {
45025             if(!s.length){
45026                 return;
45027             }
45028             
45029             Roo.get(_this.iframe.contentDocument.head).createChild({
45030                 tag : 'link',
45031                 rel : 'stylesheet',
45032                 type : 'text/css',
45033                 href : s
45034             });
45035         });
45036
45037         
45038     },
45039     
45040     removeStylesheets : function()
45041     {
45042         var _this = this;
45043         
45044         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
45045             s.remove();
45046         });
45047     },
45048     
45049     setStyle : function(style)
45050     {
45051         Roo.get(this.iframe.contentDocument.head).createChild({
45052             tag : 'style',
45053             type : 'text/css',
45054             html : style
45055         });
45056
45057         return;
45058     }
45059     
45060     // hide stuff that is not compatible
45061     /**
45062      * @event blur
45063      * @hide
45064      */
45065     /**
45066      * @event change
45067      * @hide
45068      */
45069     /**
45070      * @event focus
45071      * @hide
45072      */
45073     /**
45074      * @event specialkey
45075      * @hide
45076      */
45077     /**
45078      * @cfg {String} fieldClass @hide
45079      */
45080     /**
45081      * @cfg {String} focusClass @hide
45082      */
45083     /**
45084      * @cfg {String} autoCreate @hide
45085      */
45086     /**
45087      * @cfg {String} inputType @hide
45088      */
45089     /**
45090      * @cfg {String} invalidClass @hide
45091      */
45092     /**
45093      * @cfg {String} invalidText @hide
45094      */
45095     /**
45096      * @cfg {String} msgFx @hide
45097      */
45098     /**
45099      * @cfg {String} validateOnBlur @hide
45100      */
45101 });
45102
45103 Roo.HtmlEditorCore.white = [
45104         'area', 'br', 'img', 'input', 'hr', 'wbr',
45105         
45106        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
45107        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
45108        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
45109        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
45110        'table',   'ul',         'xmp', 
45111        
45112        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
45113       'thead',   'tr', 
45114      
45115       'dir', 'menu', 'ol', 'ul', 'dl',
45116        
45117       'embed',  'object'
45118 ];
45119
45120
45121 Roo.HtmlEditorCore.black = [
45122     //    'embed',  'object', // enable - backend responsiblity to clean thiese
45123         'applet', // 
45124         'base',   'basefont', 'bgsound', 'blink',  'body', 
45125         'frame',  'frameset', 'head',    'html',   'ilayer', 
45126         'iframe', 'layer',  'link',     'meta',    'object',   
45127         'script', 'style' ,'title',  'xml' // clean later..
45128 ];
45129 Roo.HtmlEditorCore.clean = [
45130     'script', 'style', 'title', 'xml'
45131 ];
45132 Roo.HtmlEditorCore.remove = [
45133     'font'
45134 ];
45135 // attributes..
45136
45137 Roo.HtmlEditorCore.ablack = [
45138     'on'
45139 ];
45140     
45141 Roo.HtmlEditorCore.aclean = [ 
45142     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
45143 ];
45144
45145 // protocols..
45146 Roo.HtmlEditorCore.pwhite= [
45147         'http',  'https',  'mailto'
45148 ];
45149
45150 // white listed style attributes.
45151 Roo.HtmlEditorCore.cwhite= [
45152       //  'text-align', /// default is to allow most things..
45153       
45154          
45155 //        'font-size'//??
45156 ];
45157
45158 // black listed style attributes.
45159 Roo.HtmlEditorCore.cblack= [
45160       //  'font-size' -- this can be set by the project 
45161 ];
45162
45163
45164 Roo.HtmlEditorCore.swapCodes   =[ 
45165     [    8211, "--" ], 
45166     [    8212, "--" ], 
45167     [    8216,  "'" ],  
45168     [    8217, "'" ],  
45169     [    8220, '"' ],  
45170     [    8221, '"' ],  
45171     [    8226, "*" ],  
45172     [    8230, "..." ]
45173 ]; 
45174
45175     //<script type="text/javascript">
45176
45177 /*
45178  * Ext JS Library 1.1.1
45179  * Copyright(c) 2006-2007, Ext JS, LLC.
45180  * Licence LGPL
45181  * 
45182  */
45183  
45184  
45185 Roo.form.HtmlEditor = function(config){
45186     
45187     
45188     
45189     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45190     
45191     if (!this.toolbars) {
45192         this.toolbars = [];
45193     }
45194     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45195     
45196     
45197 };
45198
45199 /**
45200  * @class Roo.form.HtmlEditor
45201  * @extends Roo.form.Field
45202  * Provides a lightweight HTML Editor component.
45203  *
45204  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45205  * 
45206  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45207  * supported by this editor.</b><br/><br/>
45208  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45209  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45210  */
45211 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45212     /**
45213      * @cfg {Boolean} clearUp
45214      */
45215     clearUp : true,
45216       /**
45217      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45218      */
45219     toolbars : false,
45220    
45221      /**
45222      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45223      *                        Roo.resizable.
45224      */
45225     resizable : false,
45226      /**
45227      * @cfg {Number} height (in pixels)
45228      */   
45229     height: 300,
45230    /**
45231      * @cfg {Number} width (in pixels)
45232      */   
45233     width: 500,
45234     
45235     /**
45236      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45237      * 
45238      */
45239     stylesheets: false,
45240     
45241     
45242      /**
45243      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45244      * 
45245      */
45246     cblack: false,
45247     /**
45248      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45249      * 
45250      */
45251     cwhite: false,
45252     
45253      /**
45254      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45255      * 
45256      */
45257     black: false,
45258     /**
45259      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45260      * 
45261      */
45262     white: false,
45263     
45264     // id of frame..
45265     frameId: false,
45266     
45267     // private properties
45268     validationEvent : false,
45269     deferHeight: true,
45270     initialized : false,
45271     activated : false,
45272     
45273     onFocus : Roo.emptyFn,
45274     iframePad:3,
45275     hideMode:'offsets',
45276     
45277     actionMode : 'container', // defaults to hiding it...
45278     
45279     defaultAutoCreate : { // modified by initCompnoent..
45280         tag: "textarea",
45281         style:"width:500px;height:300px;",
45282         autocomplete: "new-password"
45283     },
45284
45285     // private
45286     initComponent : function(){
45287         this.addEvents({
45288             /**
45289              * @event initialize
45290              * Fires when the editor is fully initialized (including the iframe)
45291              * @param {HtmlEditor} this
45292              */
45293             initialize: true,
45294             /**
45295              * @event activate
45296              * Fires when the editor is first receives the focus. Any insertion must wait
45297              * until after this event.
45298              * @param {HtmlEditor} this
45299              */
45300             activate: true,
45301              /**
45302              * @event beforesync
45303              * Fires before the textarea is updated with content from the editor iframe. Return false
45304              * to cancel the sync.
45305              * @param {HtmlEditor} this
45306              * @param {String} html
45307              */
45308             beforesync: true,
45309              /**
45310              * @event beforepush
45311              * Fires before the iframe editor is updated with content from the textarea. Return false
45312              * to cancel the push.
45313              * @param {HtmlEditor} this
45314              * @param {String} html
45315              */
45316             beforepush: true,
45317              /**
45318              * @event sync
45319              * Fires when the textarea is updated with content from the editor iframe.
45320              * @param {HtmlEditor} this
45321              * @param {String} html
45322              */
45323             sync: true,
45324              /**
45325              * @event push
45326              * Fires when the iframe editor is updated with content from the textarea.
45327              * @param {HtmlEditor} this
45328              * @param {String} html
45329              */
45330             push: true,
45331              /**
45332              * @event editmodechange
45333              * Fires when the editor switches edit modes
45334              * @param {HtmlEditor} this
45335              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45336              */
45337             editmodechange: true,
45338             /**
45339              * @event editorevent
45340              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45341              * @param {HtmlEditor} this
45342              */
45343             editorevent: true,
45344             /**
45345              * @event firstfocus
45346              * Fires when on first focus - needed by toolbars..
45347              * @param {HtmlEditor} this
45348              */
45349             firstfocus: true,
45350             /**
45351              * @event autosave
45352              * Auto save the htmlEditor value as a file into Events
45353              * @param {HtmlEditor} this
45354              */
45355             autosave: true,
45356             /**
45357              * @event savedpreview
45358              * preview the saved version of htmlEditor
45359              * @param {HtmlEditor} this
45360              */
45361             savedpreview: true,
45362             
45363             /**
45364             * @event stylesheetsclick
45365             * Fires when press the Sytlesheets button
45366             * @param {Roo.HtmlEditorCore} this
45367             */
45368             stylesheetsclick: true
45369         });
45370         this.defaultAutoCreate =  {
45371             tag: "textarea",
45372             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45373             autocomplete: "new-password"
45374         };
45375     },
45376
45377     /**
45378      * Protected method that will not generally be called directly. It
45379      * is called when the editor creates its toolbar. Override this method if you need to
45380      * add custom toolbar buttons.
45381      * @param {HtmlEditor} editor
45382      */
45383     createToolbar : function(editor){
45384         Roo.log("create toolbars");
45385         if (!editor.toolbars || !editor.toolbars.length) {
45386             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45387         }
45388         
45389         for (var i =0 ; i < editor.toolbars.length;i++) {
45390             editor.toolbars[i] = Roo.factory(
45391                     typeof(editor.toolbars[i]) == 'string' ?
45392                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45393                 Roo.form.HtmlEditor);
45394             editor.toolbars[i].init(editor);
45395         }
45396          
45397         
45398     },
45399
45400      
45401     // private
45402     onRender : function(ct, position)
45403     {
45404         var _t = this;
45405         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45406         
45407         this.wrap = this.el.wrap({
45408             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45409         });
45410         
45411         this.editorcore.onRender(ct, position);
45412          
45413         if (this.resizable) {
45414             this.resizeEl = new Roo.Resizable(this.wrap, {
45415                 pinned : true,
45416                 wrap: true,
45417                 dynamic : true,
45418                 minHeight : this.height,
45419                 height: this.height,
45420                 handles : this.resizable,
45421                 width: this.width,
45422                 listeners : {
45423                     resize : function(r, w, h) {
45424                         _t.onResize(w,h); // -something
45425                     }
45426                 }
45427             });
45428             
45429         }
45430         this.createToolbar(this);
45431        
45432         
45433         if(!this.width){
45434             this.setSize(this.wrap.getSize());
45435         }
45436         if (this.resizeEl) {
45437             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45438             // should trigger onReize..
45439         }
45440         
45441         this.keyNav = new Roo.KeyNav(this.el, {
45442             
45443             "tab" : function(e){
45444                 e.preventDefault();
45445                 
45446                 var value = this.getValue();
45447                 
45448                 var start = this.el.dom.selectionStart;
45449                 var end = this.el.dom.selectionEnd;
45450                 
45451                 if(!e.shiftKey){
45452                     
45453                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45454                     this.el.dom.setSelectionRange(end + 1, end + 1);
45455                     return;
45456                 }
45457                 
45458                 var f = value.substring(0, start).split("\t");
45459                 
45460                 if(f.pop().length != 0){
45461                     return;
45462                 }
45463                 
45464                 this.setValue(f.join("\t") + value.substring(end));
45465                 this.el.dom.setSelectionRange(start - 1, start - 1);
45466                 
45467             },
45468             
45469             "home" : function(e){
45470                 e.preventDefault();
45471                 
45472                 var curr = this.el.dom.selectionStart;
45473                 var lines = this.getValue().split("\n");
45474                 
45475                 if(!lines.length){
45476                     return;
45477                 }
45478                 
45479                 if(e.ctrlKey){
45480                     this.el.dom.setSelectionRange(0, 0);
45481                     return;
45482                 }
45483                 
45484                 var pos = 0;
45485                 
45486                 for (var i = 0; i < lines.length;i++) {
45487                     pos += lines[i].length;
45488                     
45489                     if(i != 0){
45490                         pos += 1;
45491                     }
45492                     
45493                     if(pos < curr){
45494                         continue;
45495                     }
45496                     
45497                     pos -= lines[i].length;
45498                     
45499                     break;
45500                 }
45501                 
45502                 if(!e.shiftKey){
45503                     this.el.dom.setSelectionRange(pos, pos);
45504                     return;
45505                 }
45506                 
45507                 this.el.dom.selectionStart = pos;
45508                 this.el.dom.selectionEnd = curr;
45509             },
45510             
45511             "end" : function(e){
45512                 e.preventDefault();
45513                 
45514                 var curr = this.el.dom.selectionStart;
45515                 var lines = this.getValue().split("\n");
45516                 
45517                 if(!lines.length){
45518                     return;
45519                 }
45520                 
45521                 if(e.ctrlKey){
45522                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45523                     return;
45524                 }
45525                 
45526                 var pos = 0;
45527                 
45528                 for (var i = 0; i < lines.length;i++) {
45529                     
45530                     pos += lines[i].length;
45531                     
45532                     if(i != 0){
45533                         pos += 1;
45534                     }
45535                     
45536                     if(pos < curr){
45537                         continue;
45538                     }
45539                     
45540                     break;
45541                 }
45542                 
45543                 if(!e.shiftKey){
45544                     this.el.dom.setSelectionRange(pos, pos);
45545                     return;
45546                 }
45547                 
45548                 this.el.dom.selectionStart = curr;
45549                 this.el.dom.selectionEnd = pos;
45550             },
45551
45552             scope : this,
45553
45554             doRelay : function(foo, bar, hname){
45555                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45556             },
45557
45558             forceKeyDown: true
45559         });
45560         
45561 //        if(this.autosave && this.w){
45562 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45563 //        }
45564     },
45565
45566     // private
45567     onResize : function(w, h)
45568     {
45569         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45570         var ew = false;
45571         var eh = false;
45572         
45573         if(this.el ){
45574             if(typeof w == 'number'){
45575                 var aw = w - this.wrap.getFrameWidth('lr');
45576                 this.el.setWidth(this.adjustWidth('textarea', aw));
45577                 ew = aw;
45578             }
45579             if(typeof h == 'number'){
45580                 var tbh = 0;
45581                 for (var i =0; i < this.toolbars.length;i++) {
45582                     // fixme - ask toolbars for heights?
45583                     tbh += this.toolbars[i].tb.el.getHeight();
45584                     if (this.toolbars[i].footer) {
45585                         tbh += this.toolbars[i].footer.el.getHeight();
45586                     }
45587                 }
45588                 
45589                 
45590                 
45591                 
45592                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45593                 ah -= 5; // knock a few pixes off for look..
45594 //                Roo.log(ah);
45595                 this.el.setHeight(this.adjustWidth('textarea', ah));
45596                 var eh = ah;
45597             }
45598         }
45599         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45600         this.editorcore.onResize(ew,eh);
45601         
45602     },
45603
45604     /**
45605      * Toggles the editor between standard and source edit mode.
45606      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45607      */
45608     toggleSourceEdit : function(sourceEditMode)
45609     {
45610         this.editorcore.toggleSourceEdit(sourceEditMode);
45611         
45612         if(this.editorcore.sourceEditMode){
45613             Roo.log('editor - showing textarea');
45614             
45615 //            Roo.log('in');
45616 //            Roo.log(this.syncValue());
45617             this.editorcore.syncValue();
45618             this.el.removeClass('x-hidden');
45619             this.el.dom.removeAttribute('tabIndex');
45620             this.el.focus();
45621             
45622             for (var i = 0; i < this.toolbars.length; i++) {
45623                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45624                     this.toolbars[i].tb.hide();
45625                     this.toolbars[i].footer.hide();
45626                 }
45627             }
45628             
45629         }else{
45630             Roo.log('editor - hiding textarea');
45631 //            Roo.log('out')
45632 //            Roo.log(this.pushValue()); 
45633             this.editorcore.pushValue();
45634             
45635             this.el.addClass('x-hidden');
45636             this.el.dom.setAttribute('tabIndex', -1);
45637             
45638             for (var i = 0; i < this.toolbars.length; i++) {
45639                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45640                     this.toolbars[i].tb.show();
45641                     this.toolbars[i].footer.show();
45642                 }
45643             }
45644             
45645             //this.deferFocus();
45646         }
45647         
45648         this.setSize(this.wrap.getSize());
45649         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45650         
45651         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45652     },
45653  
45654     // private (for BoxComponent)
45655     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45656
45657     // private (for BoxComponent)
45658     getResizeEl : function(){
45659         return this.wrap;
45660     },
45661
45662     // private (for BoxComponent)
45663     getPositionEl : function(){
45664         return this.wrap;
45665     },
45666
45667     // private
45668     initEvents : function(){
45669         this.originalValue = this.getValue();
45670     },
45671
45672     /**
45673      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45674      * @method
45675      */
45676     markInvalid : Roo.emptyFn,
45677     /**
45678      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45679      * @method
45680      */
45681     clearInvalid : Roo.emptyFn,
45682
45683     setValue : function(v){
45684         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45685         this.editorcore.pushValue();
45686     },
45687
45688      
45689     // private
45690     deferFocus : function(){
45691         this.focus.defer(10, this);
45692     },
45693
45694     // doc'ed in Field
45695     focus : function(){
45696         this.editorcore.focus();
45697         
45698     },
45699       
45700
45701     // private
45702     onDestroy : function(){
45703         
45704         
45705         
45706         if(this.rendered){
45707             
45708             for (var i =0; i < this.toolbars.length;i++) {
45709                 // fixme - ask toolbars for heights?
45710                 this.toolbars[i].onDestroy();
45711             }
45712             
45713             this.wrap.dom.innerHTML = '';
45714             this.wrap.remove();
45715         }
45716     },
45717
45718     // private
45719     onFirstFocus : function(){
45720         //Roo.log("onFirstFocus");
45721         this.editorcore.onFirstFocus();
45722          for (var i =0; i < this.toolbars.length;i++) {
45723             this.toolbars[i].onFirstFocus();
45724         }
45725         
45726     },
45727     
45728     // private
45729     syncValue : function()
45730     {
45731         this.editorcore.syncValue();
45732     },
45733     
45734     pushValue : function()
45735     {
45736         this.editorcore.pushValue();
45737     },
45738     
45739     setStylesheets : function(stylesheets)
45740     {
45741         this.editorcore.setStylesheets(stylesheets);
45742     },
45743     
45744     removeStylesheets : function()
45745     {
45746         this.editorcore.removeStylesheets();
45747     }
45748      
45749     
45750     // hide stuff that is not compatible
45751     /**
45752      * @event blur
45753      * @hide
45754      */
45755     /**
45756      * @event change
45757      * @hide
45758      */
45759     /**
45760      * @event focus
45761      * @hide
45762      */
45763     /**
45764      * @event specialkey
45765      * @hide
45766      */
45767     /**
45768      * @cfg {String} fieldClass @hide
45769      */
45770     /**
45771      * @cfg {String} focusClass @hide
45772      */
45773     /**
45774      * @cfg {String} autoCreate @hide
45775      */
45776     /**
45777      * @cfg {String} inputType @hide
45778      */
45779     /**
45780      * @cfg {String} invalidClass @hide
45781      */
45782     /**
45783      * @cfg {String} invalidText @hide
45784      */
45785     /**
45786      * @cfg {String} msgFx @hide
45787      */
45788     /**
45789      * @cfg {String} validateOnBlur @hide
45790      */
45791 });
45792  
45793     // <script type="text/javascript">
45794 /*
45795  * Based on
45796  * Ext JS Library 1.1.1
45797  * Copyright(c) 2006-2007, Ext JS, LLC.
45798  *  
45799  
45800  */
45801
45802 /**
45803  * @class Roo.form.HtmlEditorToolbar1
45804  * Basic Toolbar
45805  * 
45806  * Usage:
45807  *
45808  new Roo.form.HtmlEditor({
45809     ....
45810     toolbars : [
45811         new Roo.form.HtmlEditorToolbar1({
45812             disable : { fonts: 1 , format: 1, ..., ... , ...],
45813             btns : [ .... ]
45814         })
45815     }
45816      
45817  * 
45818  * @cfg {Object} disable List of elements to disable..
45819  * @cfg {Array} btns List of additional buttons.
45820  * 
45821  * 
45822  * NEEDS Extra CSS? 
45823  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45824  */
45825  
45826 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45827 {
45828     
45829     Roo.apply(this, config);
45830     
45831     // default disabled, based on 'good practice'..
45832     this.disable = this.disable || {};
45833     Roo.applyIf(this.disable, {
45834         fontSize : true,
45835         colors : true,
45836         specialElements : true
45837     });
45838     
45839     
45840     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45841     // dont call parent... till later.
45842 }
45843
45844 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45845     
45846     tb: false,
45847     
45848     rendered: false,
45849     
45850     editor : false,
45851     editorcore : false,
45852     /**
45853      * @cfg {Object} disable  List of toolbar elements to disable
45854          
45855      */
45856     disable : false,
45857     
45858     
45859      /**
45860      * @cfg {String} createLinkText The default text for the create link prompt
45861      */
45862     createLinkText : 'Please enter the URL for the link:',
45863     /**
45864      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45865      */
45866     defaultLinkValue : 'http:/'+'/',
45867    
45868     
45869       /**
45870      * @cfg {Array} fontFamilies An array of available font families
45871      */
45872     fontFamilies : [
45873         'Arial',
45874         'Courier New',
45875         'Tahoma',
45876         'Times New Roman',
45877         'Verdana'
45878     ],
45879     
45880     specialChars : [
45881            "&#169;",
45882           "&#174;",     
45883           "&#8482;",    
45884           "&#163;" ,    
45885          // "&#8212;",    
45886           "&#8230;",    
45887           "&#247;" ,    
45888         //  "&#225;" ,     ?? a acute?
45889            "&#8364;"    , //Euro
45890        //   "&#8220;"    ,
45891         //  "&#8221;"    ,
45892         //  "&#8226;"    ,
45893           "&#176;"  //   , // degrees
45894
45895          // "&#233;"     , // e ecute
45896          // "&#250;"     , // u ecute?
45897     ],
45898     
45899     specialElements : [
45900         {
45901             text: "Insert Table",
45902             xtype: 'MenuItem',
45903             xns : Roo.Menu,
45904             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45905                 
45906         },
45907         {    
45908             text: "Insert Image",
45909             xtype: 'MenuItem',
45910             xns : Roo.Menu,
45911             ihtml : '<img src="about:blank"/>'
45912             
45913         }
45914         
45915          
45916     ],
45917     
45918     
45919     inputElements : [ 
45920             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45921             "input:submit", "input:button", "select", "textarea", "label" ],
45922     formats : [
45923         ["p"] ,  
45924         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45925         ["pre"],[ "code"], 
45926         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45927         ['div'],['span'],
45928         ['sup'],['sub']
45929     ],
45930     
45931     cleanStyles : [
45932         "font-size"
45933     ],
45934      /**
45935      * @cfg {String} defaultFont default font to use.
45936      */
45937     defaultFont: 'tahoma',
45938    
45939     fontSelect : false,
45940     
45941     
45942     formatCombo : false,
45943     
45944     init : function(editor)
45945     {
45946         this.editor = editor;
45947         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45948         var editorcore = this.editorcore;
45949         
45950         var _t = this;
45951         
45952         var fid = editorcore.frameId;
45953         var etb = this;
45954         function btn(id, toggle, handler){
45955             var xid = fid + '-'+ id ;
45956             return {
45957                 id : xid,
45958                 cmd : id,
45959                 cls : 'x-btn-icon x-edit-'+id,
45960                 enableToggle:toggle !== false,
45961                 scope: _t, // was editor...
45962                 handler:handler||_t.relayBtnCmd,
45963                 clickEvent:'mousedown',
45964                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45965                 tabIndex:-1
45966             };
45967         }
45968         
45969         
45970         
45971         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45972         this.tb = tb;
45973          // stop form submits
45974         tb.el.on('click', function(e){
45975             e.preventDefault(); // what does this do?
45976         });
45977
45978         if(!this.disable.font) { // && !Roo.isSafari){
45979             /* why no safari for fonts 
45980             editor.fontSelect = tb.el.createChild({
45981                 tag:'select',
45982                 tabIndex: -1,
45983                 cls:'x-font-select',
45984                 html: this.createFontOptions()
45985             });
45986             
45987             editor.fontSelect.on('change', function(){
45988                 var font = editor.fontSelect.dom.value;
45989                 editor.relayCmd('fontname', font);
45990                 editor.deferFocus();
45991             }, editor);
45992             
45993             tb.add(
45994                 editor.fontSelect.dom,
45995                 '-'
45996             );
45997             */
45998             
45999         };
46000         if(!this.disable.formats){
46001             this.formatCombo = new Roo.form.ComboBox({
46002                 store: new Roo.data.SimpleStore({
46003                     id : 'tag',
46004                     fields: ['tag'],
46005                     data : this.formats // from states.js
46006                 }),
46007                 blockFocus : true,
46008                 name : '',
46009                 //autoCreate : {tag: "div",  size: "20"},
46010                 displayField:'tag',
46011                 typeAhead: false,
46012                 mode: 'local',
46013                 editable : false,
46014                 triggerAction: 'all',
46015                 emptyText:'Add tag',
46016                 selectOnFocus:true,
46017                 width:135,
46018                 listeners : {
46019                     'select': function(c, r, i) {
46020                         editorcore.insertTag(r.get('tag'));
46021                         editor.focus();
46022                     }
46023                 }
46024
46025             });
46026             tb.addField(this.formatCombo);
46027             
46028         }
46029         
46030         if(!this.disable.format){
46031             tb.add(
46032                 btn('bold'),
46033                 btn('italic'),
46034                 btn('underline'),
46035                 btn('strikethrough')
46036             );
46037         };
46038         if(!this.disable.fontSize){
46039             tb.add(
46040                 '-',
46041                 
46042                 
46043                 btn('increasefontsize', false, editorcore.adjustFont),
46044                 btn('decreasefontsize', false, editorcore.adjustFont)
46045             );
46046         };
46047         
46048         
46049         if(!this.disable.colors){
46050             tb.add(
46051                 '-', {
46052                     id:editorcore.frameId +'-forecolor',
46053                     cls:'x-btn-icon x-edit-forecolor',
46054                     clickEvent:'mousedown',
46055                     tooltip: this.buttonTips['forecolor'] || undefined,
46056                     tabIndex:-1,
46057                     menu : new Roo.menu.ColorMenu({
46058                         allowReselect: true,
46059                         focus: Roo.emptyFn,
46060                         value:'000000',
46061                         plain:true,
46062                         selectHandler: function(cp, color){
46063                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
46064                             editor.deferFocus();
46065                         },
46066                         scope: editorcore,
46067                         clickEvent:'mousedown'
46068                     })
46069                 }, {
46070                     id:editorcore.frameId +'backcolor',
46071                     cls:'x-btn-icon x-edit-backcolor',
46072                     clickEvent:'mousedown',
46073                     tooltip: this.buttonTips['backcolor'] || undefined,
46074                     tabIndex:-1,
46075                     menu : new Roo.menu.ColorMenu({
46076                         focus: Roo.emptyFn,
46077                         value:'FFFFFF',
46078                         plain:true,
46079                         allowReselect: true,
46080                         selectHandler: function(cp, color){
46081                             if(Roo.isGecko){
46082                                 editorcore.execCmd('useCSS', false);
46083                                 editorcore.execCmd('hilitecolor', color);
46084                                 editorcore.execCmd('useCSS', true);
46085                                 editor.deferFocus();
46086                             }else{
46087                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
46088                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
46089                                 editor.deferFocus();
46090                             }
46091                         },
46092                         scope:editorcore,
46093                         clickEvent:'mousedown'
46094                     })
46095                 }
46096             );
46097         };
46098         // now add all the items...
46099         
46100
46101         if(!this.disable.alignments){
46102             tb.add(
46103                 '-',
46104                 btn('justifyleft'),
46105                 btn('justifycenter'),
46106                 btn('justifyright')
46107             );
46108         };
46109
46110         //if(!Roo.isSafari){
46111             if(!this.disable.links){
46112                 tb.add(
46113                     '-',
46114                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
46115                 );
46116             };
46117
46118             if(!this.disable.lists){
46119                 tb.add(
46120                     '-',
46121                     btn('insertorderedlist'),
46122                     btn('insertunorderedlist')
46123                 );
46124             }
46125             if(!this.disable.sourceEdit){
46126                 tb.add(
46127                     '-',
46128                     btn('sourceedit', true, function(btn){
46129                         this.toggleSourceEdit(btn.pressed);
46130                     })
46131                 );
46132             }
46133         //}
46134         
46135         var smenu = { };
46136         // special menu.. - needs to be tidied up..
46137         if (!this.disable.special) {
46138             smenu = {
46139                 text: "&#169;",
46140                 cls: 'x-edit-none',
46141                 
46142                 menu : {
46143                     items : []
46144                 }
46145             };
46146             for (var i =0; i < this.specialChars.length; i++) {
46147                 smenu.menu.items.push({
46148                     
46149                     html: this.specialChars[i],
46150                     handler: function(a,b) {
46151                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
46152                         //editor.insertAtCursor(a.html);
46153                         
46154                     },
46155                     tabIndex:-1
46156                 });
46157             }
46158             
46159             
46160             tb.add(smenu);
46161             
46162             
46163         }
46164         
46165         var cmenu = { };
46166         if (!this.disable.cleanStyles) {
46167             cmenu = {
46168                 cls: 'x-btn-icon x-btn-clear',
46169                 
46170                 menu : {
46171                     items : []
46172                 }
46173             };
46174             for (var i =0; i < this.cleanStyles.length; i++) {
46175                 cmenu.menu.items.push({
46176                     actiontype : this.cleanStyles[i],
46177                     html: 'Remove ' + this.cleanStyles[i],
46178                     handler: function(a,b) {
46179 //                        Roo.log(a);
46180 //                        Roo.log(b);
46181                         var c = Roo.get(editorcore.doc.body);
46182                         c.select('[style]').each(function(s) {
46183                             s.dom.style.removeProperty(a.actiontype);
46184                         });
46185                         editorcore.syncValue();
46186                     },
46187                     tabIndex:-1
46188                 });
46189             }
46190              cmenu.menu.items.push({
46191                 actiontype : 'tablewidths',
46192                 html: 'Remove Table Widths',
46193                 handler: function(a,b) {
46194                     editorcore.cleanTableWidths();
46195                     editorcore.syncValue();
46196                 },
46197                 tabIndex:-1
46198             });
46199             cmenu.menu.items.push({
46200                 actiontype : 'word',
46201                 html: 'Remove MS Word Formating',
46202                 handler: function(a,b) {
46203                     editorcore.cleanWord();
46204                     editorcore.syncValue();
46205                 },
46206                 tabIndex:-1
46207             });
46208             
46209             cmenu.menu.items.push({
46210                 actiontype : 'all',
46211                 html: 'Remove All Styles',
46212                 handler: function(a,b) {
46213                     
46214                     var c = Roo.get(editorcore.doc.body);
46215                     c.select('[style]').each(function(s) {
46216                         s.dom.removeAttribute('style');
46217                     });
46218                     editorcore.syncValue();
46219                 },
46220                 tabIndex:-1
46221             });
46222             
46223             cmenu.menu.items.push({
46224                 actiontype : 'all',
46225                 html: 'Remove All CSS Classes',
46226                 handler: function(a,b) {
46227                     
46228                     var c = Roo.get(editorcore.doc.body);
46229                     c.select('[class]').each(function(s) {
46230                         s.dom.removeAttribute('class');
46231                     });
46232                     editorcore.cleanWord();
46233                     editorcore.syncValue();
46234                 },
46235                 tabIndex:-1
46236             });
46237             
46238              cmenu.menu.items.push({
46239                 actiontype : 'tidy',
46240                 html: 'Tidy HTML Source',
46241                 handler: function(a,b) {
46242                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46243                     editorcore.syncValue();
46244                 },
46245                 tabIndex:-1
46246             });
46247             
46248             
46249             tb.add(cmenu);
46250         }
46251          
46252         if (!this.disable.specialElements) {
46253             var semenu = {
46254                 text: "Other;",
46255                 cls: 'x-edit-none',
46256                 menu : {
46257                     items : []
46258                 }
46259             };
46260             for (var i =0; i < this.specialElements.length; i++) {
46261                 semenu.menu.items.push(
46262                     Roo.apply({ 
46263                         handler: function(a,b) {
46264                             editor.insertAtCursor(this.ihtml);
46265                         }
46266                     }, this.specialElements[i])
46267                 );
46268                     
46269             }
46270             
46271             tb.add(semenu);
46272             
46273             
46274         }
46275          
46276         
46277         if (this.btns) {
46278             for(var i =0; i< this.btns.length;i++) {
46279                 var b = Roo.factory(this.btns[i],Roo.form);
46280                 b.cls =  'x-edit-none';
46281                 
46282                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46283                     b.cls += ' x-init-enable';
46284                 }
46285                 
46286                 b.scope = editorcore;
46287                 tb.add(b);
46288             }
46289         
46290         }
46291         
46292         
46293         
46294         // disable everything...
46295         
46296         this.tb.items.each(function(item){
46297             
46298            if(
46299                 item.id != editorcore.frameId+ '-sourceedit' && 
46300                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46301             ){
46302                 
46303                 item.disable();
46304             }
46305         });
46306         this.rendered = true;
46307         
46308         // the all the btns;
46309         editor.on('editorevent', this.updateToolbar, this);
46310         // other toolbars need to implement this..
46311         //editor.on('editmodechange', this.updateToolbar, this);
46312     },
46313     
46314     
46315     relayBtnCmd : function(btn) {
46316         this.editorcore.relayCmd(btn.cmd);
46317     },
46318     // private used internally
46319     createLink : function(){
46320         Roo.log("create link?");
46321         var url = prompt(this.createLinkText, this.defaultLinkValue);
46322         if(url && url != 'http:/'+'/'){
46323             this.editorcore.relayCmd('createlink', url);
46324         }
46325     },
46326
46327     
46328     /**
46329      * Protected method that will not generally be called directly. It triggers
46330      * a toolbar update by reading the markup state of the current selection in the editor.
46331      */
46332     updateToolbar: function(){
46333
46334         if(!this.editorcore.activated){
46335             this.editor.onFirstFocus();
46336             return;
46337         }
46338
46339         var btns = this.tb.items.map, 
46340             doc = this.editorcore.doc,
46341             frameId = this.editorcore.frameId;
46342
46343         if(!this.disable.font && !Roo.isSafari){
46344             /*
46345             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46346             if(name != this.fontSelect.dom.value){
46347                 this.fontSelect.dom.value = name;
46348             }
46349             */
46350         }
46351         if(!this.disable.format){
46352             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46353             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46354             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46355             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46356         }
46357         if(!this.disable.alignments){
46358             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46359             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46360             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46361         }
46362         if(!Roo.isSafari && !this.disable.lists){
46363             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46364             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46365         }
46366         
46367         var ans = this.editorcore.getAllAncestors();
46368         if (this.formatCombo) {
46369             
46370             
46371             var store = this.formatCombo.store;
46372             this.formatCombo.setValue("");
46373             for (var i =0; i < ans.length;i++) {
46374                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46375                     // select it..
46376                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46377                     break;
46378                 }
46379             }
46380         }
46381         
46382         
46383         
46384         // hides menus... - so this cant be on a menu...
46385         Roo.menu.MenuMgr.hideAll();
46386
46387         //this.editorsyncValue();
46388     },
46389    
46390     
46391     createFontOptions : function(){
46392         var buf = [], fs = this.fontFamilies, ff, lc;
46393         
46394         
46395         
46396         for(var i = 0, len = fs.length; i< len; i++){
46397             ff = fs[i];
46398             lc = ff.toLowerCase();
46399             buf.push(
46400                 '<option value="',lc,'" style="font-family:',ff,';"',
46401                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46402                     ff,
46403                 '</option>'
46404             );
46405         }
46406         return buf.join('');
46407     },
46408     
46409     toggleSourceEdit : function(sourceEditMode){
46410         
46411         Roo.log("toolbar toogle");
46412         if(sourceEditMode === undefined){
46413             sourceEditMode = !this.sourceEditMode;
46414         }
46415         this.sourceEditMode = sourceEditMode === true;
46416         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46417         // just toggle the button?
46418         if(btn.pressed !== this.sourceEditMode){
46419             btn.toggle(this.sourceEditMode);
46420             return;
46421         }
46422         
46423         if(sourceEditMode){
46424             Roo.log("disabling buttons");
46425             this.tb.items.each(function(item){
46426                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46427                     item.disable();
46428                 }
46429             });
46430           
46431         }else{
46432             Roo.log("enabling buttons");
46433             if(this.editorcore.initialized){
46434                 this.tb.items.each(function(item){
46435                     item.enable();
46436                 });
46437             }
46438             
46439         }
46440         Roo.log("calling toggole on editor");
46441         // tell the editor that it's been pressed..
46442         this.editor.toggleSourceEdit(sourceEditMode);
46443        
46444     },
46445      /**
46446      * Object collection of toolbar tooltips for the buttons in the editor. The key
46447      * is the command id associated with that button and the value is a valid QuickTips object.
46448      * For example:
46449 <pre><code>
46450 {
46451     bold : {
46452         title: 'Bold (Ctrl+B)',
46453         text: 'Make the selected text bold.',
46454         cls: 'x-html-editor-tip'
46455     },
46456     italic : {
46457         title: 'Italic (Ctrl+I)',
46458         text: 'Make the selected text italic.',
46459         cls: 'x-html-editor-tip'
46460     },
46461     ...
46462 </code></pre>
46463     * @type Object
46464      */
46465     buttonTips : {
46466         bold : {
46467             title: 'Bold (Ctrl+B)',
46468             text: 'Make the selected text bold.',
46469             cls: 'x-html-editor-tip'
46470         },
46471         italic : {
46472             title: 'Italic (Ctrl+I)',
46473             text: 'Make the selected text italic.',
46474             cls: 'x-html-editor-tip'
46475         },
46476         underline : {
46477             title: 'Underline (Ctrl+U)',
46478             text: 'Underline the selected text.',
46479             cls: 'x-html-editor-tip'
46480         },
46481         strikethrough : {
46482             title: 'Strikethrough',
46483             text: 'Strikethrough the selected text.',
46484             cls: 'x-html-editor-tip'
46485         },
46486         increasefontsize : {
46487             title: 'Grow Text',
46488             text: 'Increase the font size.',
46489             cls: 'x-html-editor-tip'
46490         },
46491         decreasefontsize : {
46492             title: 'Shrink Text',
46493             text: 'Decrease the font size.',
46494             cls: 'x-html-editor-tip'
46495         },
46496         backcolor : {
46497             title: 'Text Highlight Color',
46498             text: 'Change the background color of the selected text.',
46499             cls: 'x-html-editor-tip'
46500         },
46501         forecolor : {
46502             title: 'Font Color',
46503             text: 'Change the color of the selected text.',
46504             cls: 'x-html-editor-tip'
46505         },
46506         justifyleft : {
46507             title: 'Align Text Left',
46508             text: 'Align text to the left.',
46509             cls: 'x-html-editor-tip'
46510         },
46511         justifycenter : {
46512             title: 'Center Text',
46513             text: 'Center text in the editor.',
46514             cls: 'x-html-editor-tip'
46515         },
46516         justifyright : {
46517             title: 'Align Text Right',
46518             text: 'Align text to the right.',
46519             cls: 'x-html-editor-tip'
46520         },
46521         insertunorderedlist : {
46522             title: 'Bullet List',
46523             text: 'Start a bulleted list.',
46524             cls: 'x-html-editor-tip'
46525         },
46526         insertorderedlist : {
46527             title: 'Numbered List',
46528             text: 'Start a numbered list.',
46529             cls: 'x-html-editor-tip'
46530         },
46531         createlink : {
46532             title: 'Hyperlink',
46533             text: 'Make the selected text a hyperlink.',
46534             cls: 'x-html-editor-tip'
46535         },
46536         sourceedit : {
46537             title: 'Source Edit',
46538             text: 'Switch to source editing mode.',
46539             cls: 'x-html-editor-tip'
46540         }
46541     },
46542     // private
46543     onDestroy : function(){
46544         if(this.rendered){
46545             
46546             this.tb.items.each(function(item){
46547                 if(item.menu){
46548                     item.menu.removeAll();
46549                     if(item.menu.el){
46550                         item.menu.el.destroy();
46551                     }
46552                 }
46553                 item.destroy();
46554             });
46555              
46556         }
46557     },
46558     onFirstFocus: function() {
46559         this.tb.items.each(function(item){
46560            item.enable();
46561         });
46562     }
46563 });
46564
46565
46566
46567
46568 // <script type="text/javascript">
46569 /*
46570  * Based on
46571  * Ext JS Library 1.1.1
46572  * Copyright(c) 2006-2007, Ext JS, LLC.
46573  *  
46574  
46575  */
46576
46577  
46578 /**
46579  * @class Roo.form.HtmlEditor.ToolbarContext
46580  * Context Toolbar
46581  * 
46582  * Usage:
46583  *
46584  new Roo.form.HtmlEditor({
46585     ....
46586     toolbars : [
46587         { xtype: 'ToolbarStandard', styles : {} }
46588         { xtype: 'ToolbarContext', disable : {} }
46589     ]
46590 })
46591
46592      
46593  * 
46594  * @config : {Object} disable List of elements to disable.. (not done yet.)
46595  * @config : {Object} styles  Map of styles available.
46596  * 
46597  */
46598
46599 Roo.form.HtmlEditor.ToolbarContext = function(config)
46600 {
46601     
46602     Roo.apply(this, config);
46603     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46604     // dont call parent... till later.
46605     this.styles = this.styles || {};
46606 }
46607
46608  
46609
46610 Roo.form.HtmlEditor.ToolbarContext.types = {
46611     'IMG' : {
46612         width : {
46613             title: "Width",
46614             width: 40
46615         },
46616         height:  {
46617             title: "Height",
46618             width: 40
46619         },
46620         align: {
46621             title: "Align",
46622             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46623             width : 80
46624             
46625         },
46626         border: {
46627             title: "Border",
46628             width: 40
46629         },
46630         alt: {
46631             title: "Alt",
46632             width: 120
46633         },
46634         src : {
46635             title: "Src",
46636             width: 220
46637         }
46638         
46639     },
46640     'A' : {
46641         name : {
46642             title: "Name",
46643             width: 50
46644         },
46645         target:  {
46646             title: "Target",
46647             width: 120
46648         },
46649         href:  {
46650             title: "Href",
46651             width: 220
46652         } // border?
46653         
46654     },
46655     'TABLE' : {
46656         rows : {
46657             title: "Rows",
46658             width: 20
46659         },
46660         cols : {
46661             title: "Cols",
46662             width: 20
46663         },
46664         width : {
46665             title: "Width",
46666             width: 40
46667         },
46668         height : {
46669             title: "Height",
46670             width: 40
46671         },
46672         border : {
46673             title: "Border",
46674             width: 20
46675         }
46676     },
46677     'TD' : {
46678         width : {
46679             title: "Width",
46680             width: 40
46681         },
46682         height : {
46683             title: "Height",
46684             width: 40
46685         },   
46686         align: {
46687             title: "Align",
46688             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46689             width: 80
46690         },
46691         valign: {
46692             title: "Valign",
46693             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46694             width: 80
46695         },
46696         colspan: {
46697             title: "Colspan",
46698             width: 20
46699             
46700         },
46701          'font-family'  : {
46702             title : "Font",
46703             style : 'fontFamily',
46704             displayField: 'display',
46705             optname : 'font-family',
46706             width: 140
46707         }
46708     },
46709     'INPUT' : {
46710         name : {
46711             title: "name",
46712             width: 120
46713         },
46714         value : {
46715             title: "Value",
46716             width: 120
46717         },
46718         width : {
46719             title: "Width",
46720             width: 40
46721         }
46722     },
46723     'LABEL' : {
46724         'for' : {
46725             title: "For",
46726             width: 120
46727         }
46728     },
46729     'TEXTAREA' : {
46730           name : {
46731             title: "name",
46732             width: 120
46733         },
46734         rows : {
46735             title: "Rows",
46736             width: 20
46737         },
46738         cols : {
46739             title: "Cols",
46740             width: 20
46741         }
46742     },
46743     'SELECT' : {
46744         name : {
46745             title: "name",
46746             width: 120
46747         },
46748         selectoptions : {
46749             title: "Options",
46750             width: 200
46751         }
46752     },
46753     
46754     // should we really allow this??
46755     // should this just be 
46756     'BODY' : {
46757         title : {
46758             title: "Title",
46759             width: 200,
46760             disabled : true
46761         }
46762     },
46763     'SPAN' : {
46764         'font-family'  : {
46765             title : "Font",
46766             style : 'fontFamily',
46767             displayField: 'display',
46768             optname : 'font-family',
46769             width: 140
46770         }
46771     },
46772     'DIV' : {
46773         'font-family'  : {
46774             title : "Font",
46775             style : 'fontFamily',
46776             displayField: 'display',
46777             optname : 'font-family',
46778             width: 140
46779         }
46780     },
46781      'P' : {
46782         'font-family'  : {
46783             title : "Font",
46784             style : 'fontFamily',
46785             displayField: 'display',
46786             optname : 'font-family',
46787             width: 140
46788         }
46789     },
46790     
46791     '*' : {
46792         // empty..
46793     }
46794
46795 };
46796
46797 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46798 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46799
46800 Roo.form.HtmlEditor.ToolbarContext.options = {
46801         'font-family'  : [ 
46802                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46803                 [ 'Courier New', 'Courier New'],
46804                 [ 'Tahoma', 'Tahoma'],
46805                 [ 'Times New Roman,serif', 'Times'],
46806                 [ 'Verdana','Verdana' ]
46807         ]
46808 };
46809
46810 // fixme - these need to be configurable..
46811  
46812
46813 //Roo.form.HtmlEditor.ToolbarContext.types
46814
46815
46816 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46817     
46818     tb: false,
46819     
46820     rendered: false,
46821     
46822     editor : false,
46823     editorcore : false,
46824     /**
46825      * @cfg {Object} disable  List of toolbar elements to disable
46826          
46827      */
46828     disable : false,
46829     /**
46830      * @cfg {Object} styles List of styles 
46831      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46832      *
46833      * These must be defined in the page, so they get rendered correctly..
46834      * .headline { }
46835      * TD.underline { }
46836      * 
46837      */
46838     styles : false,
46839     
46840     options: false,
46841     
46842     toolbars : false,
46843     
46844     init : function(editor)
46845     {
46846         this.editor = editor;
46847         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46848         var editorcore = this.editorcore;
46849         
46850         var fid = editorcore.frameId;
46851         var etb = this;
46852         function btn(id, toggle, handler){
46853             var xid = fid + '-'+ id ;
46854             return {
46855                 id : xid,
46856                 cmd : id,
46857                 cls : 'x-btn-icon x-edit-'+id,
46858                 enableToggle:toggle !== false,
46859                 scope: editorcore, // was editor...
46860                 handler:handler||editorcore.relayBtnCmd,
46861                 clickEvent:'mousedown',
46862                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46863                 tabIndex:-1
46864             };
46865         }
46866         // create a new element.
46867         var wdiv = editor.wrap.createChild({
46868                 tag: 'div'
46869             }, editor.wrap.dom.firstChild.nextSibling, true);
46870         
46871         // can we do this more than once??
46872         
46873          // stop form submits
46874       
46875  
46876         // disable everything...
46877         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46878         this.toolbars = {};
46879            
46880         for (var i in  ty) {
46881           
46882             this.toolbars[i] = this.buildToolbar(ty[i],i);
46883         }
46884         this.tb = this.toolbars.BODY;
46885         this.tb.el.show();
46886         this.buildFooter();
46887         this.footer.show();
46888         editor.on('hide', function( ) { this.footer.hide() }, this);
46889         editor.on('show', function( ) { this.footer.show() }, this);
46890         
46891          
46892         this.rendered = true;
46893         
46894         // the all the btns;
46895         editor.on('editorevent', this.updateToolbar, this);
46896         // other toolbars need to implement this..
46897         //editor.on('editmodechange', this.updateToolbar, this);
46898     },
46899     
46900     
46901     
46902     /**
46903      * Protected method that will not generally be called directly. It triggers
46904      * a toolbar update by reading the markup state of the current selection in the editor.
46905      *
46906      * Note you can force an update by calling on('editorevent', scope, false)
46907      */
46908     updateToolbar: function(editor,ev,sel){
46909
46910         //Roo.log(ev);
46911         // capture mouse up - this is handy for selecting images..
46912         // perhaps should go somewhere else...
46913         if(!this.editorcore.activated){
46914              this.editor.onFirstFocus();
46915             return;
46916         }
46917         
46918         
46919         
46920         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46921         // selectNode - might want to handle IE?
46922         if (ev &&
46923             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46924             ev.target && ev.target.tagName == 'IMG') {
46925             // they have click on an image...
46926             // let's see if we can change the selection...
46927             sel = ev.target;
46928          
46929               var nodeRange = sel.ownerDocument.createRange();
46930             try {
46931                 nodeRange.selectNode(sel);
46932             } catch (e) {
46933                 nodeRange.selectNodeContents(sel);
46934             }
46935             //nodeRange.collapse(true);
46936             var s = this.editorcore.win.getSelection();
46937             s.removeAllRanges();
46938             s.addRange(nodeRange);
46939         }  
46940         
46941       
46942         var updateFooter = sel ? false : true;
46943         
46944         
46945         var ans = this.editorcore.getAllAncestors();
46946         
46947         // pick
46948         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46949         
46950         if (!sel) { 
46951             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46952             sel = sel ? sel : this.editorcore.doc.body;
46953             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46954             
46955         }
46956         // pick a menu that exists..
46957         var tn = sel.tagName.toUpperCase();
46958         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46959         
46960         tn = sel.tagName.toUpperCase();
46961         
46962         var lastSel = this.tb.selectedNode;
46963         
46964         this.tb.selectedNode = sel;
46965         
46966         // if current menu does not match..
46967         
46968         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46969                 
46970             this.tb.el.hide();
46971             ///console.log("show: " + tn);
46972             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46973             this.tb.el.show();
46974             // update name
46975             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46976             
46977             
46978             // update attributes
46979             if (this.tb.fields) {
46980                 this.tb.fields.each(function(e) {
46981                     if (e.stylename) {
46982                         e.setValue(sel.style[e.stylename]);
46983                         return;
46984                     } 
46985                    e.setValue(sel.getAttribute(e.attrname));
46986                 });
46987             }
46988             
46989             var hasStyles = false;
46990             for(var i in this.styles) {
46991                 hasStyles = true;
46992                 break;
46993             }
46994             
46995             // update styles
46996             if (hasStyles) { 
46997                 var st = this.tb.fields.item(0);
46998                 
46999                 st.store.removeAll();
47000                
47001                 
47002                 var cn = sel.className.split(/\s+/);
47003                 
47004                 var avs = [];
47005                 if (this.styles['*']) {
47006                     
47007                     Roo.each(this.styles['*'], function(v) {
47008                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
47009                     });
47010                 }
47011                 if (this.styles[tn]) { 
47012                     Roo.each(this.styles[tn], function(v) {
47013                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
47014                     });
47015                 }
47016                 
47017                 st.store.loadData(avs);
47018                 st.collapse();
47019                 st.setValue(cn);
47020             }
47021             // flag our selected Node.
47022             this.tb.selectedNode = sel;
47023            
47024            
47025             Roo.menu.MenuMgr.hideAll();
47026
47027         }
47028         
47029         if (!updateFooter) {
47030             //this.footDisp.dom.innerHTML = ''; 
47031             return;
47032         }
47033         // update the footer
47034         //
47035         var html = '';
47036         
47037         this.footerEls = ans.reverse();
47038         Roo.each(this.footerEls, function(a,i) {
47039             if (!a) { return; }
47040             html += html.length ? ' &gt; '  :  '';
47041             
47042             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
47043             
47044         });
47045        
47046         // 
47047         var sz = this.footDisp.up('td').getSize();
47048         this.footDisp.dom.style.width = (sz.width -10) + 'px';
47049         this.footDisp.dom.style.marginLeft = '5px';
47050         
47051         this.footDisp.dom.style.overflow = 'hidden';
47052         
47053         this.footDisp.dom.innerHTML = html;
47054             
47055         //this.editorsyncValue();
47056     },
47057      
47058     
47059    
47060        
47061     // private
47062     onDestroy : function(){
47063         if(this.rendered){
47064             
47065             this.tb.items.each(function(item){
47066                 if(item.menu){
47067                     item.menu.removeAll();
47068                     if(item.menu.el){
47069                         item.menu.el.destroy();
47070                     }
47071                 }
47072                 item.destroy();
47073             });
47074              
47075         }
47076     },
47077     onFirstFocus: function() {
47078         // need to do this for all the toolbars..
47079         this.tb.items.each(function(item){
47080            item.enable();
47081         });
47082     },
47083     buildToolbar: function(tlist, nm)
47084     {
47085         var editor = this.editor;
47086         var editorcore = this.editorcore;
47087          // create a new element.
47088         var wdiv = editor.wrap.createChild({
47089                 tag: 'div'
47090             }, editor.wrap.dom.firstChild.nextSibling, true);
47091         
47092        
47093         var tb = new Roo.Toolbar(wdiv);
47094         // add the name..
47095         
47096         tb.add(nm+ ":&nbsp;");
47097         
47098         var styles = [];
47099         for(var i in this.styles) {
47100             styles.push(i);
47101         }
47102         
47103         // styles...
47104         if (styles && styles.length) {
47105             
47106             // this needs a multi-select checkbox...
47107             tb.addField( new Roo.form.ComboBox({
47108                 store: new Roo.data.SimpleStore({
47109                     id : 'val',
47110                     fields: ['val', 'selected'],
47111                     data : [] 
47112                 }),
47113                 name : '-roo-edit-className',
47114                 attrname : 'className',
47115                 displayField: 'val',
47116                 typeAhead: false,
47117                 mode: 'local',
47118                 editable : false,
47119                 triggerAction: 'all',
47120                 emptyText:'Select Style',
47121                 selectOnFocus:true,
47122                 width: 130,
47123                 listeners : {
47124                     'select': function(c, r, i) {
47125                         // initial support only for on class per el..
47126                         tb.selectedNode.className =  r ? r.get('val') : '';
47127                         editorcore.syncValue();
47128                     }
47129                 }
47130     
47131             }));
47132         }
47133         
47134         var tbc = Roo.form.HtmlEditor.ToolbarContext;
47135         var tbops = tbc.options;
47136         
47137         for (var i in tlist) {
47138             
47139             var item = tlist[i];
47140             tb.add(item.title + ":&nbsp;");
47141             
47142             
47143             //optname == used so you can configure the options available..
47144             var opts = item.opts ? item.opts : false;
47145             if (item.optname) {
47146                 opts = tbops[item.optname];
47147            
47148             }
47149             
47150             if (opts) {
47151                 // opts == pulldown..
47152                 tb.addField( new Roo.form.ComboBox({
47153                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
47154                         id : 'val',
47155                         fields: ['val', 'display'],
47156                         data : opts  
47157                     }),
47158                     name : '-roo-edit-' + i,
47159                     attrname : i,
47160                     stylename : item.style ? item.style : false,
47161                     displayField: item.displayField ? item.displayField : 'val',
47162                     valueField :  'val',
47163                     typeAhead: false,
47164                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47165                     editable : false,
47166                     triggerAction: 'all',
47167                     emptyText:'Select',
47168                     selectOnFocus:true,
47169                     width: item.width ? item.width  : 130,
47170                     listeners : {
47171                         'select': function(c, r, i) {
47172                             if (c.stylename) {
47173                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47174                                 return;
47175                             }
47176                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47177                         }
47178                     }
47179
47180                 }));
47181                 continue;
47182                     
47183                  
47184                 
47185                 tb.addField( new Roo.form.TextField({
47186                     name: i,
47187                     width: 100,
47188                     //allowBlank:false,
47189                     value: ''
47190                 }));
47191                 continue;
47192             }
47193             tb.addField( new Roo.form.TextField({
47194                 name: '-roo-edit-' + i,
47195                 attrname : i,
47196                 
47197                 width: item.width,
47198                 //allowBlank:true,
47199                 value: '',
47200                 listeners: {
47201                     'change' : function(f, nv, ov) {
47202                         tb.selectedNode.setAttribute(f.attrname, nv);
47203                         editorcore.syncValue();
47204                     }
47205                 }
47206             }));
47207              
47208         }
47209         
47210         var _this = this;
47211         
47212         if(nm == 'BODY'){
47213             tb.addSeparator();
47214         
47215             tb.addButton( {
47216                 text: 'Stylesheets',
47217
47218                 listeners : {
47219                     click : function ()
47220                     {
47221                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47222                     }
47223                 }
47224             });
47225         }
47226         
47227         tb.addFill();
47228         tb.addButton( {
47229             text: 'Remove Tag',
47230     
47231             listeners : {
47232                 click : function ()
47233                 {
47234                     // remove
47235                     // undo does not work.
47236                      
47237                     var sn = tb.selectedNode;
47238                     
47239                     var pn = sn.parentNode;
47240                     
47241                     var stn =  sn.childNodes[0];
47242                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47243                     while (sn.childNodes.length) {
47244                         var node = sn.childNodes[0];
47245                         sn.removeChild(node);
47246                         //Roo.log(node);
47247                         pn.insertBefore(node, sn);
47248                         
47249                     }
47250                     pn.removeChild(sn);
47251                     var range = editorcore.createRange();
47252         
47253                     range.setStart(stn,0);
47254                     range.setEnd(en,0); //????
47255                     //range.selectNode(sel);
47256                     
47257                     
47258                     var selection = editorcore.getSelection();
47259                     selection.removeAllRanges();
47260                     selection.addRange(range);
47261                     
47262                     
47263                     
47264                     //_this.updateToolbar(null, null, pn);
47265                     _this.updateToolbar(null, null, null);
47266                     _this.footDisp.dom.innerHTML = ''; 
47267                 }
47268             }
47269             
47270                     
47271                 
47272             
47273         });
47274         
47275         
47276         tb.el.on('click', function(e){
47277             e.preventDefault(); // what does this do?
47278         });
47279         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47280         tb.el.hide();
47281         tb.name = nm;
47282         // dont need to disable them... as they will get hidden
47283         return tb;
47284          
47285         
47286     },
47287     buildFooter : function()
47288     {
47289         
47290         var fel = this.editor.wrap.createChild();
47291         this.footer = new Roo.Toolbar(fel);
47292         // toolbar has scrolly on left / right?
47293         var footDisp= new Roo.Toolbar.Fill();
47294         var _t = this;
47295         this.footer.add(
47296             {
47297                 text : '&lt;',
47298                 xtype: 'Button',
47299                 handler : function() {
47300                     _t.footDisp.scrollTo('left',0,true)
47301                 }
47302             }
47303         );
47304         this.footer.add( footDisp );
47305         this.footer.add( 
47306             {
47307                 text : '&gt;',
47308                 xtype: 'Button',
47309                 handler : function() {
47310                     // no animation..
47311                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47312                 }
47313             }
47314         );
47315         var fel = Roo.get(footDisp.el);
47316         fel.addClass('x-editor-context');
47317         this.footDispWrap = fel; 
47318         this.footDispWrap.overflow  = 'hidden';
47319         
47320         this.footDisp = fel.createChild();
47321         this.footDispWrap.on('click', this.onContextClick, this)
47322         
47323         
47324     },
47325     onContextClick : function (ev,dom)
47326     {
47327         ev.preventDefault();
47328         var  cn = dom.className;
47329         //Roo.log(cn);
47330         if (!cn.match(/x-ed-loc-/)) {
47331             return;
47332         }
47333         var n = cn.split('-').pop();
47334         var ans = this.footerEls;
47335         var sel = ans[n];
47336         
47337          // pick
47338         var range = this.editorcore.createRange();
47339         
47340         range.selectNodeContents(sel);
47341         //range.selectNode(sel);
47342         
47343         
47344         var selection = this.editorcore.getSelection();
47345         selection.removeAllRanges();
47346         selection.addRange(range);
47347         
47348         
47349         
47350         this.updateToolbar(null, null, sel);
47351         
47352         
47353     }
47354     
47355     
47356     
47357     
47358     
47359 });
47360
47361
47362
47363
47364
47365 /*
47366  * Based on:
47367  * Ext JS Library 1.1.1
47368  * Copyright(c) 2006-2007, Ext JS, LLC.
47369  *
47370  * Originally Released Under LGPL - original licence link has changed is not relivant.
47371  *
47372  * Fork - LGPL
47373  * <script type="text/javascript">
47374  */
47375  
47376 /**
47377  * @class Roo.form.BasicForm
47378  * @extends Roo.util.Observable
47379  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47380  * @constructor
47381  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47382  * @param {Object} config Configuration options
47383  */
47384 Roo.form.BasicForm = function(el, config){
47385     this.allItems = [];
47386     this.childForms = [];
47387     Roo.apply(this, config);
47388     /*
47389      * The Roo.form.Field items in this form.
47390      * @type MixedCollection
47391      */
47392      
47393      
47394     this.items = new Roo.util.MixedCollection(false, function(o){
47395         return o.id || (o.id = Roo.id());
47396     });
47397     this.addEvents({
47398         /**
47399          * @event beforeaction
47400          * Fires before any action is performed. Return false to cancel the action.
47401          * @param {Form} this
47402          * @param {Action} action The action to be performed
47403          */
47404         beforeaction: true,
47405         /**
47406          * @event actionfailed
47407          * Fires when an action fails.
47408          * @param {Form} this
47409          * @param {Action} action The action that failed
47410          */
47411         actionfailed : true,
47412         /**
47413          * @event actioncomplete
47414          * Fires when an action is completed.
47415          * @param {Form} this
47416          * @param {Action} action The action that completed
47417          */
47418         actioncomplete : true
47419     });
47420     if(el){
47421         this.initEl(el);
47422     }
47423     Roo.form.BasicForm.superclass.constructor.call(this);
47424     
47425     Roo.form.BasicForm.popover.apply();
47426 };
47427
47428 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47429     /**
47430      * @cfg {String} method
47431      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47432      */
47433     /**
47434      * @cfg {DataReader} reader
47435      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47436      * This is optional as there is built-in support for processing JSON.
47437      */
47438     /**
47439      * @cfg {DataReader} errorReader
47440      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47441      * This is completely optional as there is built-in support for processing JSON.
47442      */
47443     /**
47444      * @cfg {String} url
47445      * The URL to use for form actions if one isn't supplied in the action options.
47446      */
47447     /**
47448      * @cfg {Boolean} fileUpload
47449      * Set to true if this form is a file upload.
47450      */
47451      
47452     /**
47453      * @cfg {Object} baseParams
47454      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47455      */
47456      /**
47457      
47458     /**
47459      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47460      */
47461     timeout: 30,
47462
47463     // private
47464     activeAction : null,
47465
47466     /**
47467      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47468      * or setValues() data instead of when the form was first created.
47469      */
47470     trackResetOnLoad : false,
47471     
47472     
47473     /**
47474      * childForms - used for multi-tab forms
47475      * @type {Array}
47476      */
47477     childForms : false,
47478     
47479     /**
47480      * allItems - full list of fields.
47481      * @type {Array}
47482      */
47483     allItems : false,
47484     
47485     /**
47486      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47487      * element by passing it or its id or mask the form itself by passing in true.
47488      * @type Mixed
47489      */
47490     waitMsgTarget : false,
47491     
47492     /**
47493      * @type Boolean
47494      */
47495     disableMask : false,
47496     
47497     /**
47498      * @cfg {Boolean} errorMask (true|false) default false
47499      */
47500     errorMask : false,
47501     
47502     /**
47503      * @cfg {Number} maskOffset Default 100
47504      */
47505     maskOffset : 100,
47506
47507     // private
47508     initEl : function(el){
47509         this.el = Roo.get(el);
47510         this.id = this.el.id || Roo.id();
47511         this.el.on('submit', this.onSubmit, this);
47512         this.el.addClass('x-form');
47513     },
47514
47515     // private
47516     onSubmit : function(e){
47517         e.stopEvent();
47518     },
47519
47520     /**
47521      * Returns true if client-side validation on the form is successful.
47522      * @return Boolean
47523      */
47524     isValid : function(){
47525         var valid = true;
47526         var target = false;
47527         this.items.each(function(f){
47528             if(f.validate()){
47529                 return;
47530             }
47531             
47532             valid = false;
47533                 
47534             if(!target && f.el.isVisible(true)){
47535                 target = f;
47536             }
47537         });
47538         
47539         if(this.errorMask && !valid){
47540             Roo.form.BasicForm.popover.mask(this, target);
47541         }
47542         
47543         return valid;
47544     },
47545
47546     /**
47547      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47548      * @return Boolean
47549      */
47550     isDirty : function(){
47551         var dirty = false;
47552         this.items.each(function(f){
47553            if(f.isDirty()){
47554                dirty = true;
47555                return false;
47556            }
47557         });
47558         return dirty;
47559     },
47560     
47561     /**
47562      * Returns true if any fields in this form have changed since their original load. (New version)
47563      * @return Boolean
47564      */
47565     
47566     hasChanged : function()
47567     {
47568         var dirty = false;
47569         this.items.each(function(f){
47570            if(f.hasChanged()){
47571                dirty = true;
47572                return false;
47573            }
47574         });
47575         return dirty;
47576         
47577     },
47578     /**
47579      * Resets all hasChanged to 'false' -
47580      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47581      * So hasChanged storage is only to be used for this purpose
47582      * @return Boolean
47583      */
47584     resetHasChanged : function()
47585     {
47586         this.items.each(function(f){
47587            f.resetHasChanged();
47588         });
47589         
47590     },
47591     
47592     
47593     /**
47594      * Performs a predefined action (submit or load) or custom actions you define on this form.
47595      * @param {String} actionName The name of the action type
47596      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47597      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47598      * accept other config options):
47599      * <pre>
47600 Property          Type             Description
47601 ----------------  ---------------  ----------------------------------------------------------------------------------
47602 url               String           The url for the action (defaults to the form's url)
47603 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47604 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47605 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47606                                    validate the form on the client (defaults to false)
47607      * </pre>
47608      * @return {BasicForm} this
47609      */
47610     doAction : function(action, options){
47611         if(typeof action == 'string'){
47612             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47613         }
47614         if(this.fireEvent('beforeaction', this, action) !== false){
47615             this.beforeAction(action);
47616             action.run.defer(100, action);
47617         }
47618         return this;
47619     },
47620
47621     /**
47622      * Shortcut to do a submit action.
47623      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47624      * @return {BasicForm} this
47625      */
47626     submit : function(options){
47627         this.doAction('submit', options);
47628         return this;
47629     },
47630
47631     /**
47632      * Shortcut to do a load action.
47633      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47634      * @return {BasicForm} this
47635      */
47636     load : function(options){
47637         this.doAction('load', options);
47638         return this;
47639     },
47640
47641     /**
47642      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47643      * @param {Record} record The record to edit
47644      * @return {BasicForm} this
47645      */
47646     updateRecord : function(record){
47647         record.beginEdit();
47648         var fs = record.fields;
47649         fs.each(function(f){
47650             var field = this.findField(f.name);
47651             if(field){
47652                 record.set(f.name, field.getValue());
47653             }
47654         }, this);
47655         record.endEdit();
47656         return this;
47657     },
47658
47659     /**
47660      * Loads an Roo.data.Record into this form.
47661      * @param {Record} record The record to load
47662      * @return {BasicForm} this
47663      */
47664     loadRecord : function(record){
47665         this.setValues(record.data);
47666         return this;
47667     },
47668
47669     // private
47670     beforeAction : function(action){
47671         var o = action.options;
47672         
47673         if(!this.disableMask) {
47674             if(this.waitMsgTarget === true){
47675                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47676             }else if(this.waitMsgTarget){
47677                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47678                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47679             }else {
47680                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47681             }
47682         }
47683         
47684          
47685     },
47686
47687     // private
47688     afterAction : function(action, success){
47689         this.activeAction = null;
47690         var o = action.options;
47691         
47692         if(!this.disableMask) {
47693             if(this.waitMsgTarget === true){
47694                 this.el.unmask();
47695             }else if(this.waitMsgTarget){
47696                 this.waitMsgTarget.unmask();
47697             }else{
47698                 Roo.MessageBox.updateProgress(1);
47699                 Roo.MessageBox.hide();
47700             }
47701         }
47702         
47703         if(success){
47704             if(o.reset){
47705                 this.reset();
47706             }
47707             Roo.callback(o.success, o.scope, [this, action]);
47708             this.fireEvent('actioncomplete', this, action);
47709             
47710         }else{
47711             
47712             // failure condition..
47713             // we have a scenario where updates need confirming.
47714             // eg. if a locking scenario exists..
47715             // we look for { errors : { needs_confirm : true }} in the response.
47716             if (
47717                 (typeof(action.result) != 'undefined')  &&
47718                 (typeof(action.result.errors) != 'undefined')  &&
47719                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47720            ){
47721                 var _t = this;
47722                 Roo.MessageBox.confirm(
47723                     "Change requires confirmation",
47724                     action.result.errorMsg,
47725                     function(r) {
47726                         if (r != 'yes') {
47727                             return;
47728                         }
47729                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47730                     }
47731                     
47732                 );
47733                 
47734                 
47735                 
47736                 return;
47737             }
47738             
47739             Roo.callback(o.failure, o.scope, [this, action]);
47740             // show an error message if no failed handler is set..
47741             if (!this.hasListener('actionfailed')) {
47742                 Roo.MessageBox.alert("Error",
47743                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47744                         action.result.errorMsg :
47745                         "Saving Failed, please check your entries or try again"
47746                 );
47747             }
47748             
47749             this.fireEvent('actionfailed', this, action);
47750         }
47751         
47752     },
47753
47754     /**
47755      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47756      * @param {String} id The value to search for
47757      * @return Field
47758      */
47759     findField : function(id){
47760         var field = this.items.get(id);
47761         if(!field){
47762             this.items.each(function(f){
47763                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47764                     field = f;
47765                     return false;
47766                 }
47767             });
47768         }
47769         return field || null;
47770     },
47771
47772     /**
47773      * Add a secondary form to this one, 
47774      * Used to provide tabbed forms. One form is primary, with hidden values 
47775      * which mirror the elements from the other forms.
47776      * 
47777      * @param {Roo.form.Form} form to add.
47778      * 
47779      */
47780     addForm : function(form)
47781     {
47782        
47783         if (this.childForms.indexOf(form) > -1) {
47784             // already added..
47785             return;
47786         }
47787         this.childForms.push(form);
47788         var n = '';
47789         Roo.each(form.allItems, function (fe) {
47790             
47791             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47792             if (this.findField(n)) { // already added..
47793                 return;
47794             }
47795             var add = new Roo.form.Hidden({
47796                 name : n
47797             });
47798             add.render(this.el);
47799             
47800             this.add( add );
47801         }, this);
47802         
47803     },
47804     /**
47805      * Mark fields in this form invalid in bulk.
47806      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47807      * @return {BasicForm} this
47808      */
47809     markInvalid : function(errors){
47810         if(errors instanceof Array){
47811             for(var i = 0, len = errors.length; i < len; i++){
47812                 var fieldError = errors[i];
47813                 var f = this.findField(fieldError.id);
47814                 if(f){
47815                     f.markInvalid(fieldError.msg);
47816                 }
47817             }
47818         }else{
47819             var field, id;
47820             for(id in errors){
47821                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47822                     field.markInvalid(errors[id]);
47823                 }
47824             }
47825         }
47826         Roo.each(this.childForms || [], function (f) {
47827             f.markInvalid(errors);
47828         });
47829         
47830         return this;
47831     },
47832
47833     /**
47834      * Set values for fields in this form in bulk.
47835      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47836      * @return {BasicForm} this
47837      */
47838     setValues : function(values){
47839         if(values instanceof Array){ // array of objects
47840             for(var i = 0, len = values.length; i < len; i++){
47841                 var v = values[i];
47842                 var f = this.findField(v.id);
47843                 if(f){
47844                     f.setValue(v.value);
47845                     if(this.trackResetOnLoad){
47846                         f.originalValue = f.getValue();
47847                     }
47848                 }
47849             }
47850         }else{ // object hash
47851             var field, id;
47852             for(id in values){
47853                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47854                     
47855                     if (field.setFromData && 
47856                         field.valueField && 
47857                         field.displayField &&
47858                         // combos' with local stores can 
47859                         // be queried via setValue()
47860                         // to set their value..
47861                         (field.store && !field.store.isLocal)
47862                         ) {
47863                         // it's a combo
47864                         var sd = { };
47865                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47866                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47867                         field.setFromData(sd);
47868                         
47869                     } else {
47870                         field.setValue(values[id]);
47871                     }
47872                     
47873                     
47874                     if(this.trackResetOnLoad){
47875                         field.originalValue = field.getValue();
47876                     }
47877                 }
47878             }
47879         }
47880         this.resetHasChanged();
47881         
47882         
47883         Roo.each(this.childForms || [], function (f) {
47884             f.setValues(values);
47885             f.resetHasChanged();
47886         });
47887                 
47888         return this;
47889     },
47890  
47891     /**
47892      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47893      * they are returned as an array.
47894      * @param {Boolean} asString
47895      * @return {Object}
47896      */
47897     getValues : function(asString){
47898         if (this.childForms) {
47899             // copy values from the child forms
47900             Roo.each(this.childForms, function (f) {
47901                 this.setValues(f.getValues());
47902             }, this);
47903         }
47904         
47905         // use formdata
47906         if (typeof(FormData) != 'undefined' && asString !== true) {
47907             // this relies on a 'recent' version of chrome apparently...
47908             try {
47909                 var fd = (new FormData(this.el.dom)).entries();
47910                 var ret = {};
47911                 var ent = fd.next();
47912                 while (!ent.done) {
47913                     ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47914                     ent = fd.next();
47915                 };
47916                 return ret;
47917             } catch(e) {
47918                 
47919             }
47920             
47921         }
47922         
47923         
47924         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47925         if(asString === true){
47926             return fs;
47927         }
47928         return Roo.urlDecode(fs);
47929     },
47930     
47931     /**
47932      * Returns the fields in this form as an object with key/value pairs. 
47933      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47934      * @return {Object}
47935      */
47936     getFieldValues : function(with_hidden)
47937     {
47938         if (this.childForms) {
47939             // copy values from the child forms
47940             // should this call getFieldValues - probably not as we do not currently copy
47941             // hidden fields when we generate..
47942             Roo.each(this.childForms, function (f) {
47943                 this.setValues(f.getValues());
47944             }, this);
47945         }
47946         
47947         var ret = {};
47948         this.items.each(function(f){
47949             if (!f.getName()) {
47950                 return;
47951             }
47952             var v = f.getValue();
47953             if (f.inputType =='radio') {
47954                 if (typeof(ret[f.getName()]) == 'undefined') {
47955                     ret[f.getName()] = ''; // empty..
47956                 }
47957                 
47958                 if (!f.el.dom.checked) {
47959                     return;
47960                     
47961                 }
47962                 v = f.el.dom.value;
47963                 
47964             }
47965             
47966             // not sure if this supported any more..
47967             if ((typeof(v) == 'object') && f.getRawValue) {
47968                 v = f.getRawValue() ; // dates..
47969             }
47970             // combo boxes where name != hiddenName...
47971             if (f.name != f.getName()) {
47972                 ret[f.name] = f.getRawValue();
47973             }
47974             ret[f.getName()] = v;
47975         });
47976         
47977         return ret;
47978     },
47979
47980     /**
47981      * Clears all invalid messages in this form.
47982      * @return {BasicForm} this
47983      */
47984     clearInvalid : function(){
47985         this.items.each(function(f){
47986            f.clearInvalid();
47987         });
47988         
47989         Roo.each(this.childForms || [], function (f) {
47990             f.clearInvalid();
47991         });
47992         
47993         
47994         return this;
47995     },
47996
47997     /**
47998      * Resets this form.
47999      * @return {BasicForm} this
48000      */
48001     reset : function(){
48002         this.items.each(function(f){
48003             f.reset();
48004         });
48005         
48006         Roo.each(this.childForms || [], function (f) {
48007             f.reset();
48008         });
48009         this.resetHasChanged();
48010         
48011         return this;
48012     },
48013
48014     /**
48015      * Add Roo.form components to this form.
48016      * @param {Field} field1
48017      * @param {Field} field2 (optional)
48018      * @param {Field} etc (optional)
48019      * @return {BasicForm} this
48020      */
48021     add : function(){
48022         this.items.addAll(Array.prototype.slice.call(arguments, 0));
48023         return this;
48024     },
48025
48026
48027     /**
48028      * Removes a field from the items collection (does NOT remove its markup).
48029      * @param {Field} field
48030      * @return {BasicForm} this
48031      */
48032     remove : function(field){
48033         this.items.remove(field);
48034         return this;
48035     },
48036
48037     /**
48038      * Looks at the fields in this form, checks them for an id attribute,
48039      * and calls applyTo on the existing dom element with that id.
48040      * @return {BasicForm} this
48041      */
48042     render : function(){
48043         this.items.each(function(f){
48044             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
48045                 f.applyTo(f.id);
48046             }
48047         });
48048         return this;
48049     },
48050
48051     /**
48052      * Calls {@link Ext#apply} for all fields in this form with the passed object.
48053      * @param {Object} values
48054      * @return {BasicForm} this
48055      */
48056     applyToFields : function(o){
48057         this.items.each(function(f){
48058            Roo.apply(f, o);
48059         });
48060         return this;
48061     },
48062
48063     /**
48064      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
48065      * @param {Object} values
48066      * @return {BasicForm} this
48067      */
48068     applyIfToFields : function(o){
48069         this.items.each(function(f){
48070            Roo.applyIf(f, o);
48071         });
48072         return this;
48073     }
48074 });
48075
48076 // back compat
48077 Roo.BasicForm = Roo.form.BasicForm;
48078
48079 Roo.apply(Roo.form.BasicForm, {
48080     
48081     popover : {
48082         
48083         padding : 5,
48084         
48085         isApplied : false,
48086         
48087         isMasked : false,
48088         
48089         form : false,
48090         
48091         target : false,
48092         
48093         intervalID : false,
48094         
48095         maskEl : false,
48096         
48097         apply : function()
48098         {
48099             if(this.isApplied){
48100                 return;
48101             }
48102             
48103             this.maskEl = {
48104                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
48105                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
48106                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
48107                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
48108             };
48109             
48110             this.maskEl.top.enableDisplayMode("block");
48111             this.maskEl.left.enableDisplayMode("block");
48112             this.maskEl.bottom.enableDisplayMode("block");
48113             this.maskEl.right.enableDisplayMode("block");
48114             
48115             Roo.get(document.body).on('click', function(){
48116                 this.unmask();
48117             }, this);
48118             
48119             Roo.get(document.body).on('touchstart', function(){
48120                 this.unmask();
48121             }, this);
48122             
48123             this.isApplied = true
48124         },
48125         
48126         mask : function(form, target)
48127         {
48128             this.form = form;
48129             
48130             this.target = target;
48131             
48132             if(!this.form.errorMask || !target.el){
48133                 return;
48134             }
48135             
48136             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
48137             
48138             var ot = this.target.el.calcOffsetsTo(scrollable);
48139             
48140             var scrollTo = ot[1] - this.form.maskOffset;
48141             
48142             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
48143             
48144             scrollable.scrollTo('top', scrollTo);
48145             
48146             var el = this.target.wrap || this.target.el;
48147             
48148             var box = el.getBox();
48149             
48150             this.maskEl.top.setStyle('position', 'absolute');
48151             this.maskEl.top.setStyle('z-index', 10000);
48152             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
48153             this.maskEl.top.setLeft(0);
48154             this.maskEl.top.setTop(0);
48155             this.maskEl.top.show();
48156             
48157             this.maskEl.left.setStyle('position', 'absolute');
48158             this.maskEl.left.setStyle('z-index', 10000);
48159             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
48160             this.maskEl.left.setLeft(0);
48161             this.maskEl.left.setTop(box.y - this.padding);
48162             this.maskEl.left.show();
48163
48164             this.maskEl.bottom.setStyle('position', 'absolute');
48165             this.maskEl.bottom.setStyle('z-index', 10000);
48166             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48167             this.maskEl.bottom.setLeft(0);
48168             this.maskEl.bottom.setTop(box.bottom + this.padding);
48169             this.maskEl.bottom.show();
48170
48171             this.maskEl.right.setStyle('position', 'absolute');
48172             this.maskEl.right.setStyle('z-index', 10000);
48173             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48174             this.maskEl.right.setLeft(box.right + this.padding);
48175             this.maskEl.right.setTop(box.y - this.padding);
48176             this.maskEl.right.show();
48177
48178             this.intervalID = window.setInterval(function() {
48179                 Roo.form.BasicForm.popover.unmask();
48180             }, 10000);
48181
48182             window.onwheel = function(){ return false;};
48183             
48184             (function(){ this.isMasked = true; }).defer(500, this);
48185             
48186         },
48187         
48188         unmask : function()
48189         {
48190             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48191                 return;
48192             }
48193             
48194             this.maskEl.top.setStyle('position', 'absolute');
48195             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48196             this.maskEl.top.hide();
48197
48198             this.maskEl.left.setStyle('position', 'absolute');
48199             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48200             this.maskEl.left.hide();
48201
48202             this.maskEl.bottom.setStyle('position', 'absolute');
48203             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48204             this.maskEl.bottom.hide();
48205
48206             this.maskEl.right.setStyle('position', 'absolute');
48207             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48208             this.maskEl.right.hide();
48209             
48210             window.onwheel = function(){ return true;};
48211             
48212             if(this.intervalID){
48213                 window.clearInterval(this.intervalID);
48214                 this.intervalID = false;
48215             }
48216             
48217             this.isMasked = false;
48218             
48219         }
48220         
48221     }
48222     
48223 });/*
48224  * Based on:
48225  * Ext JS Library 1.1.1
48226  * Copyright(c) 2006-2007, Ext JS, LLC.
48227  *
48228  * Originally Released Under LGPL - original licence link has changed is not relivant.
48229  *
48230  * Fork - LGPL
48231  * <script type="text/javascript">
48232  */
48233
48234 /**
48235  * @class Roo.form.Form
48236  * @extends Roo.form.BasicForm
48237  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48238  * @constructor
48239  * @param {Object} config Configuration options
48240  */
48241 Roo.form.Form = function(config){
48242     var xitems =  [];
48243     if (config.items) {
48244         xitems = config.items;
48245         delete config.items;
48246     }
48247    
48248     
48249     Roo.form.Form.superclass.constructor.call(this, null, config);
48250     this.url = this.url || this.action;
48251     if(!this.root){
48252         this.root = new Roo.form.Layout(Roo.applyIf({
48253             id: Roo.id()
48254         }, config));
48255     }
48256     this.active = this.root;
48257     /**
48258      * Array of all the buttons that have been added to this form via {@link addButton}
48259      * @type Array
48260      */
48261     this.buttons = [];
48262     this.allItems = [];
48263     this.addEvents({
48264         /**
48265          * @event clientvalidation
48266          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48267          * @param {Form} this
48268          * @param {Boolean} valid true if the form has passed client-side validation
48269          */
48270         clientvalidation: true,
48271         /**
48272          * @event rendered
48273          * Fires when the form is rendered
48274          * @param {Roo.form.Form} form
48275          */
48276         rendered : true
48277     });
48278     
48279     if (this.progressUrl) {
48280             // push a hidden field onto the list of fields..
48281             this.addxtype( {
48282                     xns: Roo.form, 
48283                     xtype : 'Hidden', 
48284                     name : 'UPLOAD_IDENTIFIER' 
48285             });
48286         }
48287         
48288     
48289     Roo.each(xitems, this.addxtype, this);
48290     
48291 };
48292
48293 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48294     /**
48295      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48296      */
48297     /**
48298      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48299      */
48300     /**
48301      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48302      */
48303     buttonAlign:'center',
48304
48305     /**
48306      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48307      */
48308     minButtonWidth:75,
48309
48310     /**
48311      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48312      * This property cascades to child containers if not set.
48313      */
48314     labelAlign:'left',
48315
48316     /**
48317      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48318      * fires a looping event with that state. This is required to bind buttons to the valid
48319      * state using the config value formBind:true on the button.
48320      */
48321     monitorValid : false,
48322
48323     /**
48324      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48325      */
48326     monitorPoll : 200,
48327     
48328     /**
48329      * @cfg {String} progressUrl - Url to return progress data 
48330      */
48331     
48332     progressUrl : false,
48333     /**
48334      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48335      * sending a formdata with extra parameters - eg uploaded elements.
48336      */
48337     
48338     formData : false,
48339     
48340     /**
48341      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48342      * fields are added and the column is closed. If no fields are passed the column remains open
48343      * until end() is called.
48344      * @param {Object} config The config to pass to the column
48345      * @param {Field} field1 (optional)
48346      * @param {Field} field2 (optional)
48347      * @param {Field} etc (optional)
48348      * @return Column The column container object
48349      */
48350     column : function(c){
48351         var col = new Roo.form.Column(c);
48352         this.start(col);
48353         if(arguments.length > 1){ // duplicate code required because of Opera
48354             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48355             this.end();
48356         }
48357         return col;
48358     },
48359
48360     /**
48361      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48362      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48363      * until end() is called.
48364      * @param {Object} config The config to pass to the fieldset
48365      * @param {Field} field1 (optional)
48366      * @param {Field} field2 (optional)
48367      * @param {Field} etc (optional)
48368      * @return FieldSet The fieldset container object
48369      */
48370     fieldset : function(c){
48371         var fs = new Roo.form.FieldSet(c);
48372         this.start(fs);
48373         if(arguments.length > 1){ // duplicate code required because of Opera
48374             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48375             this.end();
48376         }
48377         return fs;
48378     },
48379
48380     /**
48381      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48382      * fields are added and the container is closed. If no fields are passed the container remains open
48383      * until end() is called.
48384      * @param {Object} config The config to pass to the Layout
48385      * @param {Field} field1 (optional)
48386      * @param {Field} field2 (optional)
48387      * @param {Field} etc (optional)
48388      * @return Layout The container object
48389      */
48390     container : function(c){
48391         var l = new Roo.form.Layout(c);
48392         this.start(l);
48393         if(arguments.length > 1){ // duplicate code required because of Opera
48394             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48395             this.end();
48396         }
48397         return l;
48398     },
48399
48400     /**
48401      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48402      * @param {Object} container A Roo.form.Layout or subclass of Layout
48403      * @return {Form} this
48404      */
48405     start : function(c){
48406         // cascade label info
48407         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48408         this.active.stack.push(c);
48409         c.ownerCt = this.active;
48410         this.active = c;
48411         return this;
48412     },
48413
48414     /**
48415      * Closes the current open container
48416      * @return {Form} this
48417      */
48418     end : function(){
48419         if(this.active == this.root){
48420             return this;
48421         }
48422         this.active = this.active.ownerCt;
48423         return this;
48424     },
48425
48426     /**
48427      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48428      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48429      * as the label of the field.
48430      * @param {Field} field1
48431      * @param {Field} field2 (optional)
48432      * @param {Field} etc. (optional)
48433      * @return {Form} this
48434      */
48435     add : function(){
48436         this.active.stack.push.apply(this.active.stack, arguments);
48437         this.allItems.push.apply(this.allItems,arguments);
48438         var r = [];
48439         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48440             if(a[i].isFormField){
48441                 r.push(a[i]);
48442             }
48443         }
48444         if(r.length > 0){
48445             Roo.form.Form.superclass.add.apply(this, r);
48446         }
48447         return this;
48448     },
48449     
48450
48451     
48452     
48453     
48454      /**
48455      * Find any element that has been added to a form, using it's ID or name
48456      * This can include framesets, columns etc. along with regular fields..
48457      * @param {String} id - id or name to find.
48458      
48459      * @return {Element} e - or false if nothing found.
48460      */
48461     findbyId : function(id)
48462     {
48463         var ret = false;
48464         if (!id) {
48465             return ret;
48466         }
48467         Roo.each(this.allItems, function(f){
48468             if (f.id == id || f.name == id ){
48469                 ret = f;
48470                 return false;
48471             }
48472         });
48473         return ret;
48474     },
48475
48476     
48477     
48478     /**
48479      * Render this form into the passed container. This should only be called once!
48480      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48481      * @return {Form} this
48482      */
48483     render : function(ct)
48484     {
48485         
48486         
48487         
48488         ct = Roo.get(ct);
48489         var o = this.autoCreate || {
48490             tag: 'form',
48491             method : this.method || 'POST',
48492             id : this.id || Roo.id()
48493         };
48494         this.initEl(ct.createChild(o));
48495
48496         this.root.render(this.el);
48497         
48498        
48499              
48500         this.items.each(function(f){
48501             f.render('x-form-el-'+f.id);
48502         });
48503
48504         if(this.buttons.length > 0){
48505             // tables are required to maintain order and for correct IE layout
48506             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48507                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48508                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48509             }}, null, true);
48510             var tr = tb.getElementsByTagName('tr')[0];
48511             for(var i = 0, len = this.buttons.length; i < len; i++) {
48512                 var b = this.buttons[i];
48513                 var td = document.createElement('td');
48514                 td.className = 'x-form-btn-td';
48515                 b.render(tr.appendChild(td));
48516             }
48517         }
48518         if(this.monitorValid){ // initialize after render
48519             this.startMonitoring();
48520         }
48521         this.fireEvent('rendered', this);
48522         return this;
48523     },
48524
48525     /**
48526      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48527      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48528      * object or a valid Roo.DomHelper element config
48529      * @param {Function} handler The function called when the button is clicked
48530      * @param {Object} scope (optional) The scope of the handler function
48531      * @return {Roo.Button}
48532      */
48533     addButton : function(config, handler, scope){
48534         var bc = {
48535             handler: handler,
48536             scope: scope,
48537             minWidth: this.minButtonWidth,
48538             hideParent:true
48539         };
48540         if(typeof config == "string"){
48541             bc.text = config;
48542         }else{
48543             Roo.apply(bc, config);
48544         }
48545         var btn = new Roo.Button(null, bc);
48546         this.buttons.push(btn);
48547         return btn;
48548     },
48549
48550      /**
48551      * Adds a series of form elements (using the xtype property as the factory method.
48552      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48553      * @param {Object} config 
48554      */
48555     
48556     addxtype : function()
48557     {
48558         var ar = Array.prototype.slice.call(arguments, 0);
48559         var ret = false;
48560         for(var i = 0; i < ar.length; i++) {
48561             if (!ar[i]) {
48562                 continue; // skip -- if this happends something invalid got sent, we 
48563                 // should ignore it, as basically that interface element will not show up
48564                 // and that should be pretty obvious!!
48565             }
48566             
48567             if (Roo.form[ar[i].xtype]) {
48568                 ar[i].form = this;
48569                 var fe = Roo.factory(ar[i], Roo.form);
48570                 if (!ret) {
48571                     ret = fe;
48572                 }
48573                 fe.form = this;
48574                 if (fe.store) {
48575                     fe.store.form = this;
48576                 }
48577                 if (fe.isLayout) {  
48578                          
48579                     this.start(fe);
48580                     this.allItems.push(fe);
48581                     if (fe.items && fe.addxtype) {
48582                         fe.addxtype.apply(fe, fe.items);
48583                         delete fe.items;
48584                     }
48585                      this.end();
48586                     continue;
48587                 }
48588                 
48589                 
48590                  
48591                 this.add(fe);
48592               //  console.log('adding ' + ar[i].xtype);
48593             }
48594             if (ar[i].xtype == 'Button') {  
48595                 //console.log('adding button');
48596                 //console.log(ar[i]);
48597                 this.addButton(ar[i]);
48598                 this.allItems.push(fe);
48599                 continue;
48600             }
48601             
48602             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48603                 alert('end is not supported on xtype any more, use items');
48604             //    this.end();
48605             //    //console.log('adding end');
48606             }
48607             
48608         }
48609         return ret;
48610     },
48611     
48612     /**
48613      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48614      * option "monitorValid"
48615      */
48616     startMonitoring : function(){
48617         if(!this.bound){
48618             this.bound = true;
48619             Roo.TaskMgr.start({
48620                 run : this.bindHandler,
48621                 interval : this.monitorPoll || 200,
48622                 scope: this
48623             });
48624         }
48625     },
48626
48627     /**
48628      * Stops monitoring of the valid state of this form
48629      */
48630     stopMonitoring : function(){
48631         this.bound = false;
48632     },
48633
48634     // private
48635     bindHandler : function(){
48636         if(!this.bound){
48637             return false; // stops binding
48638         }
48639         var valid = true;
48640         this.items.each(function(f){
48641             if(!f.isValid(true)){
48642                 valid = false;
48643                 return false;
48644             }
48645         });
48646         for(var i = 0, len = this.buttons.length; i < len; i++){
48647             var btn = this.buttons[i];
48648             if(btn.formBind === true && btn.disabled === valid){
48649                 btn.setDisabled(!valid);
48650             }
48651         }
48652         this.fireEvent('clientvalidation', this, valid);
48653     }
48654     
48655     
48656     
48657     
48658     
48659     
48660     
48661     
48662 });
48663
48664
48665 // back compat
48666 Roo.Form = Roo.form.Form;
48667 /*
48668  * Based on:
48669  * Ext JS Library 1.1.1
48670  * Copyright(c) 2006-2007, Ext JS, LLC.
48671  *
48672  * Originally Released Under LGPL - original licence link has changed is not relivant.
48673  *
48674  * Fork - LGPL
48675  * <script type="text/javascript">
48676  */
48677
48678 // as we use this in bootstrap.
48679 Roo.namespace('Roo.form');
48680  /**
48681  * @class Roo.form.Action
48682  * Internal Class used to handle form actions
48683  * @constructor
48684  * @param {Roo.form.BasicForm} el The form element or its id
48685  * @param {Object} config Configuration options
48686  */
48687
48688  
48689  
48690 // define the action interface
48691 Roo.form.Action = function(form, options){
48692     this.form = form;
48693     this.options = options || {};
48694 };
48695 /**
48696  * Client Validation Failed
48697  * @const 
48698  */
48699 Roo.form.Action.CLIENT_INVALID = 'client';
48700 /**
48701  * Server Validation Failed
48702  * @const 
48703  */
48704 Roo.form.Action.SERVER_INVALID = 'server';
48705  /**
48706  * Connect to Server Failed
48707  * @const 
48708  */
48709 Roo.form.Action.CONNECT_FAILURE = 'connect';
48710 /**
48711  * Reading Data from Server Failed
48712  * @const 
48713  */
48714 Roo.form.Action.LOAD_FAILURE = 'load';
48715
48716 Roo.form.Action.prototype = {
48717     type : 'default',
48718     failureType : undefined,
48719     response : undefined,
48720     result : undefined,
48721
48722     // interface method
48723     run : function(options){
48724
48725     },
48726
48727     // interface method
48728     success : function(response){
48729
48730     },
48731
48732     // interface method
48733     handleResponse : function(response){
48734
48735     },
48736
48737     // default connection failure
48738     failure : function(response){
48739         
48740         this.response = response;
48741         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48742         this.form.afterAction(this, false);
48743     },
48744
48745     processResponse : function(response){
48746         this.response = response;
48747         if(!response.responseText){
48748             return true;
48749         }
48750         this.result = this.handleResponse(response);
48751         return this.result;
48752     },
48753
48754     // utility functions used internally
48755     getUrl : function(appendParams){
48756         var url = this.options.url || this.form.url || this.form.el.dom.action;
48757         if(appendParams){
48758             var p = this.getParams();
48759             if(p){
48760                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48761             }
48762         }
48763         return url;
48764     },
48765
48766     getMethod : function(){
48767         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48768     },
48769
48770     getParams : function(){
48771         var bp = this.form.baseParams;
48772         var p = this.options.params;
48773         if(p){
48774             if(typeof p == "object"){
48775                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48776             }else if(typeof p == 'string' && bp){
48777                 p += '&' + Roo.urlEncode(bp);
48778             }
48779         }else if(bp){
48780             p = Roo.urlEncode(bp);
48781         }
48782         return p;
48783     },
48784
48785     createCallback : function(){
48786         return {
48787             success: this.success,
48788             failure: this.failure,
48789             scope: this,
48790             timeout: (this.form.timeout*1000),
48791             upload: this.form.fileUpload ? this.success : undefined
48792         };
48793     }
48794 };
48795
48796 Roo.form.Action.Submit = function(form, options){
48797     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48798 };
48799
48800 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48801     type : 'submit',
48802
48803     haveProgress : false,
48804     uploadComplete : false,
48805     
48806     // uploadProgress indicator.
48807     uploadProgress : function()
48808     {
48809         if (!this.form.progressUrl) {
48810             return;
48811         }
48812         
48813         if (!this.haveProgress) {
48814             Roo.MessageBox.progress("Uploading", "Uploading");
48815         }
48816         if (this.uploadComplete) {
48817            Roo.MessageBox.hide();
48818            return;
48819         }
48820         
48821         this.haveProgress = true;
48822    
48823         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48824         
48825         var c = new Roo.data.Connection();
48826         c.request({
48827             url : this.form.progressUrl,
48828             params: {
48829                 id : uid
48830             },
48831             method: 'GET',
48832             success : function(req){
48833                //console.log(data);
48834                 var rdata = false;
48835                 var edata;
48836                 try  {
48837                    rdata = Roo.decode(req.responseText)
48838                 } catch (e) {
48839                     Roo.log("Invalid data from server..");
48840                     Roo.log(edata);
48841                     return;
48842                 }
48843                 if (!rdata || !rdata.success) {
48844                     Roo.log(rdata);
48845                     Roo.MessageBox.alert(Roo.encode(rdata));
48846                     return;
48847                 }
48848                 var data = rdata.data;
48849                 
48850                 if (this.uploadComplete) {
48851                    Roo.MessageBox.hide();
48852                    return;
48853                 }
48854                    
48855                 if (data){
48856                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48857                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48858                     );
48859                 }
48860                 this.uploadProgress.defer(2000,this);
48861             },
48862        
48863             failure: function(data) {
48864                 Roo.log('progress url failed ');
48865                 Roo.log(data);
48866             },
48867             scope : this
48868         });
48869            
48870     },
48871     
48872     
48873     run : function()
48874     {
48875         // run get Values on the form, so it syncs any secondary forms.
48876         this.form.getValues();
48877         
48878         var o = this.options;
48879         var method = this.getMethod();
48880         var isPost = method == 'POST';
48881         if(o.clientValidation === false || this.form.isValid()){
48882             
48883             if (this.form.progressUrl) {
48884                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48885                     (new Date() * 1) + '' + Math.random());
48886                     
48887             } 
48888             
48889             
48890             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48891                 form:this.form.el.dom,
48892                 url:this.getUrl(!isPost),
48893                 method: method,
48894                 params:isPost ? this.getParams() : null,
48895                 isUpload: this.form.fileUpload,
48896                 formData : this.form.formData
48897             }));
48898             
48899             this.uploadProgress();
48900
48901         }else if (o.clientValidation !== false){ // client validation failed
48902             this.failureType = Roo.form.Action.CLIENT_INVALID;
48903             this.form.afterAction(this, false);
48904         }
48905     },
48906
48907     success : function(response)
48908     {
48909         this.uploadComplete= true;
48910         if (this.haveProgress) {
48911             Roo.MessageBox.hide();
48912         }
48913         
48914         
48915         var result = this.processResponse(response);
48916         if(result === true || result.success){
48917             this.form.afterAction(this, true);
48918             return;
48919         }
48920         if(result.errors){
48921             this.form.markInvalid(result.errors);
48922             this.failureType = Roo.form.Action.SERVER_INVALID;
48923         }
48924         this.form.afterAction(this, false);
48925     },
48926     failure : function(response)
48927     {
48928         this.uploadComplete= true;
48929         if (this.haveProgress) {
48930             Roo.MessageBox.hide();
48931         }
48932         
48933         this.response = response;
48934         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48935         this.form.afterAction(this, false);
48936     },
48937     
48938     handleResponse : function(response){
48939         if(this.form.errorReader){
48940             var rs = this.form.errorReader.read(response);
48941             var errors = [];
48942             if(rs.records){
48943                 for(var i = 0, len = rs.records.length; i < len; i++) {
48944                     var r = rs.records[i];
48945                     errors[i] = r.data;
48946                 }
48947             }
48948             if(errors.length < 1){
48949                 errors = null;
48950             }
48951             return {
48952                 success : rs.success,
48953                 errors : errors
48954             };
48955         }
48956         var ret = false;
48957         try {
48958             ret = Roo.decode(response.responseText);
48959         } catch (e) {
48960             ret = {
48961                 success: false,
48962                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48963                 errors : []
48964             };
48965         }
48966         return ret;
48967         
48968     }
48969 });
48970
48971
48972 Roo.form.Action.Load = function(form, options){
48973     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48974     this.reader = this.form.reader;
48975 };
48976
48977 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48978     type : 'load',
48979
48980     run : function(){
48981         
48982         Roo.Ajax.request(Roo.apply(
48983                 this.createCallback(), {
48984                     method:this.getMethod(),
48985                     url:this.getUrl(false),
48986                     params:this.getParams()
48987         }));
48988     },
48989
48990     success : function(response){
48991         
48992         var result = this.processResponse(response);
48993         if(result === true || !result.success || !result.data){
48994             this.failureType = Roo.form.Action.LOAD_FAILURE;
48995             this.form.afterAction(this, false);
48996             return;
48997         }
48998         this.form.clearInvalid();
48999         this.form.setValues(result.data);
49000         this.form.afterAction(this, true);
49001     },
49002
49003     handleResponse : function(response){
49004         if(this.form.reader){
49005             var rs = this.form.reader.read(response);
49006             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
49007             return {
49008                 success : rs.success,
49009                 data : data
49010             };
49011         }
49012         return Roo.decode(response.responseText);
49013     }
49014 });
49015
49016 Roo.form.Action.ACTION_TYPES = {
49017     'load' : Roo.form.Action.Load,
49018     'submit' : Roo.form.Action.Submit
49019 };/*
49020  * Based on:
49021  * Ext JS Library 1.1.1
49022  * Copyright(c) 2006-2007, Ext JS, LLC.
49023  *
49024  * Originally Released Under LGPL - original licence link has changed is not relivant.
49025  *
49026  * Fork - LGPL
49027  * <script type="text/javascript">
49028  */
49029  
49030 /**
49031  * @class Roo.form.Layout
49032  * @extends Roo.Component
49033  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
49034  * @constructor
49035  * @param {Object} config Configuration options
49036  */
49037 Roo.form.Layout = function(config){
49038     var xitems = [];
49039     if (config.items) {
49040         xitems = config.items;
49041         delete config.items;
49042     }
49043     Roo.form.Layout.superclass.constructor.call(this, config);
49044     this.stack = [];
49045     Roo.each(xitems, this.addxtype, this);
49046      
49047 };
49048
49049 Roo.extend(Roo.form.Layout, Roo.Component, {
49050     /**
49051      * @cfg {String/Object} autoCreate
49052      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
49053      */
49054     /**
49055      * @cfg {String/Object/Function} style
49056      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
49057      * a function which returns such a specification.
49058      */
49059     /**
49060      * @cfg {String} labelAlign
49061      * Valid values are "left," "top" and "right" (defaults to "left")
49062      */
49063     /**
49064      * @cfg {Number} labelWidth
49065      * Fixed width in pixels of all field labels (defaults to undefined)
49066      */
49067     /**
49068      * @cfg {Boolean} clear
49069      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
49070      */
49071     clear : true,
49072     /**
49073      * @cfg {String} labelSeparator
49074      * The separator to use after field labels (defaults to ':')
49075      */
49076     labelSeparator : ':',
49077     /**
49078      * @cfg {Boolean} hideLabels
49079      * True to suppress the display of field labels in this layout (defaults to false)
49080      */
49081     hideLabels : false,
49082
49083     // private
49084     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
49085     
49086     isLayout : true,
49087     
49088     // private
49089     onRender : function(ct, position){
49090         if(this.el){ // from markup
49091             this.el = Roo.get(this.el);
49092         }else {  // generate
49093             var cfg = this.getAutoCreate();
49094             this.el = ct.createChild(cfg, position);
49095         }
49096         if(this.style){
49097             this.el.applyStyles(this.style);
49098         }
49099         if(this.labelAlign){
49100             this.el.addClass('x-form-label-'+this.labelAlign);
49101         }
49102         if(this.hideLabels){
49103             this.labelStyle = "display:none";
49104             this.elementStyle = "padding-left:0;";
49105         }else{
49106             if(typeof this.labelWidth == 'number'){
49107                 this.labelStyle = "width:"+this.labelWidth+"px;";
49108                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
49109             }
49110             if(this.labelAlign == 'top'){
49111                 this.labelStyle = "width:auto;";
49112                 this.elementStyle = "padding-left:0;";
49113             }
49114         }
49115         var stack = this.stack;
49116         var slen = stack.length;
49117         if(slen > 0){
49118             if(!this.fieldTpl){
49119                 var t = new Roo.Template(
49120                     '<div class="x-form-item {5}">',
49121                         '<label for="{0}" style="{2}">{1}{4}</label>',
49122                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49123                         '</div>',
49124                     '</div><div class="x-form-clear-left"></div>'
49125                 );
49126                 t.disableFormats = true;
49127                 t.compile();
49128                 Roo.form.Layout.prototype.fieldTpl = t;
49129             }
49130             for(var i = 0; i < slen; i++) {
49131                 if(stack[i].isFormField){
49132                     this.renderField(stack[i]);
49133                 }else{
49134                     this.renderComponent(stack[i]);
49135                 }
49136             }
49137         }
49138         if(this.clear){
49139             this.el.createChild({cls:'x-form-clear'});
49140         }
49141     },
49142
49143     // private
49144     renderField : function(f){
49145         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
49146                f.id, //0
49147                f.fieldLabel, //1
49148                f.labelStyle||this.labelStyle||'', //2
49149                this.elementStyle||'', //3
49150                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
49151                f.itemCls||this.itemCls||''  //5
49152        ], true).getPrevSibling());
49153     },
49154
49155     // private
49156     renderComponent : function(c){
49157         c.render(c.isLayout ? this.el : this.el.createChild());    
49158     },
49159     /**
49160      * Adds a object form elements (using the xtype property as the factory method.)
49161      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49162      * @param {Object} config 
49163      */
49164     addxtype : function(o)
49165     {
49166         // create the lement.
49167         o.form = this.form;
49168         var fe = Roo.factory(o, Roo.form);
49169         this.form.allItems.push(fe);
49170         this.stack.push(fe);
49171         
49172         if (fe.isFormField) {
49173             this.form.items.add(fe);
49174         }
49175          
49176         return fe;
49177     }
49178 });
49179
49180 /**
49181  * @class Roo.form.Column
49182  * @extends Roo.form.Layout
49183  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49184  * @constructor
49185  * @param {Object} config Configuration options
49186  */
49187 Roo.form.Column = function(config){
49188     Roo.form.Column.superclass.constructor.call(this, config);
49189 };
49190
49191 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49192     /**
49193      * @cfg {Number/String} width
49194      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49195      */
49196     /**
49197      * @cfg {String/Object} autoCreate
49198      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49199      */
49200
49201     // private
49202     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49203
49204     // private
49205     onRender : function(ct, position){
49206         Roo.form.Column.superclass.onRender.call(this, ct, position);
49207         if(this.width){
49208             this.el.setWidth(this.width);
49209         }
49210     }
49211 });
49212
49213
49214 /**
49215  * @class Roo.form.Row
49216  * @extends Roo.form.Layout
49217  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49218  * @constructor
49219  * @param {Object} config Configuration options
49220  */
49221
49222  
49223 Roo.form.Row = function(config){
49224     Roo.form.Row.superclass.constructor.call(this, config);
49225 };
49226  
49227 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49228       /**
49229      * @cfg {Number/String} width
49230      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49231      */
49232     /**
49233      * @cfg {Number/String} height
49234      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49235      */
49236     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49237     
49238     padWidth : 20,
49239     // private
49240     onRender : function(ct, position){
49241         //console.log('row render');
49242         if(!this.rowTpl){
49243             var t = new Roo.Template(
49244                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49245                     '<label for="{0}" style="{2}">{1}{4}</label>',
49246                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49247                     '</div>',
49248                 '</div>'
49249             );
49250             t.disableFormats = true;
49251             t.compile();
49252             Roo.form.Layout.prototype.rowTpl = t;
49253         }
49254         this.fieldTpl = this.rowTpl;
49255         
49256         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49257         var labelWidth = 100;
49258         
49259         if ((this.labelAlign != 'top')) {
49260             if (typeof this.labelWidth == 'number') {
49261                 labelWidth = this.labelWidth
49262             }
49263             this.padWidth =  20 + labelWidth;
49264             
49265         }
49266         
49267         Roo.form.Column.superclass.onRender.call(this, ct, position);
49268         if(this.width){
49269             this.el.setWidth(this.width);
49270         }
49271         if(this.height){
49272             this.el.setHeight(this.height);
49273         }
49274     },
49275     
49276     // private
49277     renderField : function(f){
49278         f.fieldEl = this.fieldTpl.append(this.el, [
49279                f.id, f.fieldLabel,
49280                f.labelStyle||this.labelStyle||'',
49281                this.elementStyle||'',
49282                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49283                f.itemCls||this.itemCls||'',
49284                f.width ? f.width + this.padWidth : 160 + this.padWidth
49285        ],true);
49286     }
49287 });
49288  
49289
49290 /**
49291  * @class Roo.form.FieldSet
49292  * @extends Roo.form.Layout
49293  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49294  * @constructor
49295  * @param {Object} config Configuration options
49296  */
49297 Roo.form.FieldSet = function(config){
49298     Roo.form.FieldSet.superclass.constructor.call(this, config);
49299 };
49300
49301 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49302     /**
49303      * @cfg {String} legend
49304      * The text to display as the legend for the FieldSet (defaults to '')
49305      */
49306     /**
49307      * @cfg {String/Object} autoCreate
49308      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49309      */
49310
49311     // private
49312     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49313
49314     // private
49315     onRender : function(ct, position){
49316         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49317         if(this.legend){
49318             this.setLegend(this.legend);
49319         }
49320     },
49321
49322     // private
49323     setLegend : function(text){
49324         if(this.rendered){
49325             this.el.child('legend').update(text);
49326         }
49327     }
49328 });/*
49329  * Based on:
49330  * Ext JS Library 1.1.1
49331  * Copyright(c) 2006-2007, Ext JS, LLC.
49332  *
49333  * Originally Released Under LGPL - original licence link has changed is not relivant.
49334  *
49335  * Fork - LGPL
49336  * <script type="text/javascript">
49337  */
49338 /**
49339  * @class Roo.form.VTypes
49340  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49341  * @singleton
49342  */
49343 Roo.form.VTypes = function(){
49344     // closure these in so they are only created once.
49345     var alpha = /^[a-zA-Z_]+$/;
49346     var alphanum = /^[a-zA-Z0-9_]+$/;
49347     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49348     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49349
49350     // All these messages and functions are configurable
49351     return {
49352         /**
49353          * The function used to validate email addresses
49354          * @param {String} value The email address
49355          */
49356         'email' : function(v){
49357             return email.test(v);
49358         },
49359         /**
49360          * The error text to display when the email validation function returns false
49361          * @type String
49362          */
49363         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49364         /**
49365          * The keystroke filter mask to be applied on email input
49366          * @type RegExp
49367          */
49368         'emailMask' : /[a-z0-9_\.\-@]/i,
49369
49370         /**
49371          * The function used to validate URLs
49372          * @param {String} value The URL
49373          */
49374         'url' : function(v){
49375             return url.test(v);
49376         },
49377         /**
49378          * The error text to display when the url validation function returns false
49379          * @type String
49380          */
49381         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49382         
49383         /**
49384          * The function used to validate alpha values
49385          * @param {String} value The value
49386          */
49387         'alpha' : function(v){
49388             return alpha.test(v);
49389         },
49390         /**
49391          * The error text to display when the alpha validation function returns false
49392          * @type String
49393          */
49394         'alphaText' : 'This field should only contain letters and _',
49395         /**
49396          * The keystroke filter mask to be applied on alpha input
49397          * @type RegExp
49398          */
49399         'alphaMask' : /[a-z_]/i,
49400
49401         /**
49402          * The function used to validate alphanumeric values
49403          * @param {String} value The value
49404          */
49405         'alphanum' : function(v){
49406             return alphanum.test(v);
49407         },
49408         /**
49409          * The error text to display when the alphanumeric validation function returns false
49410          * @type String
49411          */
49412         'alphanumText' : 'This field should only contain letters, numbers and _',
49413         /**
49414          * The keystroke filter mask to be applied on alphanumeric input
49415          * @type RegExp
49416          */
49417         'alphanumMask' : /[a-z0-9_]/i
49418     };
49419 }();//<script type="text/javascript">
49420
49421 /**
49422  * @class Roo.form.FCKeditor
49423  * @extends Roo.form.TextArea
49424  * Wrapper around the FCKEditor http://www.fckeditor.net
49425  * @constructor
49426  * Creates a new FCKeditor
49427  * @param {Object} config Configuration options
49428  */
49429 Roo.form.FCKeditor = function(config){
49430     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49431     this.addEvents({
49432          /**
49433          * @event editorinit
49434          * Fired when the editor is initialized - you can add extra handlers here..
49435          * @param {FCKeditor} this
49436          * @param {Object} the FCK object.
49437          */
49438         editorinit : true
49439     });
49440     
49441     
49442 };
49443 Roo.form.FCKeditor.editors = { };
49444 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49445 {
49446     //defaultAutoCreate : {
49447     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49448     //},
49449     // private
49450     /**
49451      * @cfg {Object} fck options - see fck manual for details.
49452      */
49453     fckconfig : false,
49454     
49455     /**
49456      * @cfg {Object} fck toolbar set (Basic or Default)
49457      */
49458     toolbarSet : 'Basic',
49459     /**
49460      * @cfg {Object} fck BasePath
49461      */ 
49462     basePath : '/fckeditor/',
49463     
49464     
49465     frame : false,
49466     
49467     value : '',
49468     
49469    
49470     onRender : function(ct, position)
49471     {
49472         if(!this.el){
49473             this.defaultAutoCreate = {
49474                 tag: "textarea",
49475                 style:"width:300px;height:60px;",
49476                 autocomplete: "new-password"
49477             };
49478         }
49479         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49480         /*
49481         if(this.grow){
49482             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49483             if(this.preventScrollbars){
49484                 this.el.setStyle("overflow", "hidden");
49485             }
49486             this.el.setHeight(this.growMin);
49487         }
49488         */
49489         //console.log('onrender' + this.getId() );
49490         Roo.form.FCKeditor.editors[this.getId()] = this;
49491          
49492
49493         this.replaceTextarea() ;
49494         
49495     },
49496     
49497     getEditor : function() {
49498         return this.fckEditor;
49499     },
49500     /**
49501      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49502      * @param {Mixed} value The value to set
49503      */
49504     
49505     
49506     setValue : function(value)
49507     {
49508         //console.log('setValue: ' + value);
49509         
49510         if(typeof(value) == 'undefined') { // not sure why this is happending...
49511             return;
49512         }
49513         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49514         
49515         //if(!this.el || !this.getEditor()) {
49516         //    this.value = value;
49517             //this.setValue.defer(100,this,[value]);    
49518         //    return;
49519         //} 
49520         
49521         if(!this.getEditor()) {
49522             return;
49523         }
49524         
49525         this.getEditor().SetData(value);
49526         
49527         //
49528
49529     },
49530
49531     /**
49532      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49533      * @return {Mixed} value The field value
49534      */
49535     getValue : function()
49536     {
49537         
49538         if (this.frame && this.frame.dom.style.display == 'none') {
49539             return Roo.form.FCKeditor.superclass.getValue.call(this);
49540         }
49541         
49542         if(!this.el || !this.getEditor()) {
49543            
49544            // this.getValue.defer(100,this); 
49545             return this.value;
49546         }
49547        
49548         
49549         var value=this.getEditor().GetData();
49550         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49551         return Roo.form.FCKeditor.superclass.getValue.call(this);
49552         
49553
49554     },
49555
49556     /**
49557      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49558      * @return {Mixed} value The field value
49559      */
49560     getRawValue : function()
49561     {
49562         if (this.frame && this.frame.dom.style.display == 'none') {
49563             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49564         }
49565         
49566         if(!this.el || !this.getEditor()) {
49567             //this.getRawValue.defer(100,this); 
49568             return this.value;
49569             return;
49570         }
49571         
49572         
49573         
49574         var value=this.getEditor().GetData();
49575         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49576         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49577          
49578     },
49579     
49580     setSize : function(w,h) {
49581         
49582         
49583         
49584         //if (this.frame && this.frame.dom.style.display == 'none') {
49585         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49586         //    return;
49587         //}
49588         //if(!this.el || !this.getEditor()) {
49589         //    this.setSize.defer(100,this, [w,h]); 
49590         //    return;
49591         //}
49592         
49593         
49594         
49595         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49596         
49597         this.frame.dom.setAttribute('width', w);
49598         this.frame.dom.setAttribute('height', h);
49599         this.frame.setSize(w,h);
49600         
49601     },
49602     
49603     toggleSourceEdit : function(value) {
49604         
49605       
49606          
49607         this.el.dom.style.display = value ? '' : 'none';
49608         this.frame.dom.style.display = value ?  'none' : '';
49609         
49610     },
49611     
49612     
49613     focus: function(tag)
49614     {
49615         if (this.frame.dom.style.display == 'none') {
49616             return Roo.form.FCKeditor.superclass.focus.call(this);
49617         }
49618         if(!this.el || !this.getEditor()) {
49619             this.focus.defer(100,this, [tag]); 
49620             return;
49621         }
49622         
49623         
49624         
49625         
49626         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49627         this.getEditor().Focus();
49628         if (tgs.length) {
49629             if (!this.getEditor().Selection.GetSelection()) {
49630                 this.focus.defer(100,this, [tag]); 
49631                 return;
49632             }
49633             
49634             
49635             var r = this.getEditor().EditorDocument.createRange();
49636             r.setStart(tgs[0],0);
49637             r.setEnd(tgs[0],0);
49638             this.getEditor().Selection.GetSelection().removeAllRanges();
49639             this.getEditor().Selection.GetSelection().addRange(r);
49640             this.getEditor().Focus();
49641         }
49642         
49643     },
49644     
49645     
49646     
49647     replaceTextarea : function()
49648     {
49649         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49650             return ;
49651         }
49652         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49653         //{
49654             // We must check the elements firstly using the Id and then the name.
49655         var oTextarea = document.getElementById( this.getId() );
49656         
49657         var colElementsByName = document.getElementsByName( this.getId() ) ;
49658          
49659         oTextarea.style.display = 'none' ;
49660
49661         if ( oTextarea.tabIndex ) {            
49662             this.TabIndex = oTextarea.tabIndex ;
49663         }
49664         
49665         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49666         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49667         this.frame = Roo.get(this.getId() + '___Frame')
49668     },
49669     
49670     _getConfigHtml : function()
49671     {
49672         var sConfig = '' ;
49673
49674         for ( var o in this.fckconfig ) {
49675             sConfig += sConfig.length > 0  ? '&amp;' : '';
49676             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49677         }
49678
49679         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49680     },
49681     
49682     
49683     _getIFrameHtml : function()
49684     {
49685         var sFile = 'fckeditor.html' ;
49686         /* no idea what this is about..
49687         try
49688         {
49689             if ( (/fcksource=true/i).test( window.top.location.search ) )
49690                 sFile = 'fckeditor.original.html' ;
49691         }
49692         catch (e) { 
49693         */
49694
49695         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49696         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49697         
49698         
49699         var html = '<iframe id="' + this.getId() +
49700             '___Frame" src="' + sLink +
49701             '" width="' + this.width +
49702             '" height="' + this.height + '"' +
49703             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49704             ' frameborder="0" scrolling="no"></iframe>' ;
49705
49706         return html ;
49707     },
49708     
49709     _insertHtmlBefore : function( html, element )
49710     {
49711         if ( element.insertAdjacentHTML )       {
49712             // IE
49713             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49714         } else { // Gecko
49715             var oRange = document.createRange() ;
49716             oRange.setStartBefore( element ) ;
49717             var oFragment = oRange.createContextualFragment( html );
49718             element.parentNode.insertBefore( oFragment, element ) ;
49719         }
49720     }
49721     
49722     
49723   
49724     
49725     
49726     
49727     
49728
49729 });
49730
49731 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49732
49733 function FCKeditor_OnComplete(editorInstance){
49734     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49735     f.fckEditor = editorInstance;
49736     //console.log("loaded");
49737     f.fireEvent('editorinit', f, editorInstance);
49738
49739   
49740
49741  
49742
49743
49744
49745
49746
49747
49748
49749
49750
49751
49752
49753
49754
49755
49756
49757 //<script type="text/javascript">
49758 /**
49759  * @class Roo.form.GridField
49760  * @extends Roo.form.Field
49761  * Embed a grid (or editable grid into a form)
49762  * STATUS ALPHA
49763  * 
49764  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49765  * it needs 
49766  * xgrid.store = Roo.data.Store
49767  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49768  * xgrid.store.reader = Roo.data.JsonReader 
49769  * 
49770  * 
49771  * @constructor
49772  * Creates a new GridField
49773  * @param {Object} config Configuration options
49774  */
49775 Roo.form.GridField = function(config){
49776     Roo.form.GridField.superclass.constructor.call(this, config);
49777      
49778 };
49779
49780 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49781     /**
49782      * @cfg {Number} width  - used to restrict width of grid..
49783      */
49784     width : 100,
49785     /**
49786      * @cfg {Number} height - used to restrict height of grid..
49787      */
49788     height : 50,
49789      /**
49790      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49791          * 
49792          *}
49793      */
49794     xgrid : false, 
49795     /**
49796      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49797      * {tag: "input", type: "checkbox", autocomplete: "off"})
49798      */
49799    // defaultAutoCreate : { tag: 'div' },
49800     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49801     /**
49802      * @cfg {String} addTitle Text to include for adding a title.
49803      */
49804     addTitle : false,
49805     //
49806     onResize : function(){
49807         Roo.form.Field.superclass.onResize.apply(this, arguments);
49808     },
49809
49810     initEvents : function(){
49811         // Roo.form.Checkbox.superclass.initEvents.call(this);
49812         // has no events...
49813        
49814     },
49815
49816
49817     getResizeEl : function(){
49818         return this.wrap;
49819     },
49820
49821     getPositionEl : function(){
49822         return this.wrap;
49823     },
49824
49825     // private
49826     onRender : function(ct, position){
49827         
49828         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49829         var style = this.style;
49830         delete this.style;
49831         
49832         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49833         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49834         this.viewEl = this.wrap.createChild({ tag: 'div' });
49835         if (style) {
49836             this.viewEl.applyStyles(style);
49837         }
49838         if (this.width) {
49839             this.viewEl.setWidth(this.width);
49840         }
49841         if (this.height) {
49842             this.viewEl.setHeight(this.height);
49843         }
49844         //if(this.inputValue !== undefined){
49845         //this.setValue(this.value);
49846         
49847         
49848         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49849         
49850         
49851         this.grid.render();
49852         this.grid.getDataSource().on('remove', this.refreshValue, this);
49853         this.grid.getDataSource().on('update', this.refreshValue, this);
49854         this.grid.on('afteredit', this.refreshValue, this);
49855  
49856     },
49857      
49858     
49859     /**
49860      * Sets the value of the item. 
49861      * @param {String} either an object  or a string..
49862      */
49863     setValue : function(v){
49864         //this.value = v;
49865         v = v || []; // empty set..
49866         // this does not seem smart - it really only affects memoryproxy grids..
49867         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49868             var ds = this.grid.getDataSource();
49869             // assumes a json reader..
49870             var data = {}
49871             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49872             ds.loadData( data);
49873         }
49874         // clear selection so it does not get stale.
49875         if (this.grid.sm) { 
49876             this.grid.sm.clearSelections();
49877         }
49878         
49879         Roo.form.GridField.superclass.setValue.call(this, v);
49880         this.refreshValue();
49881         // should load data in the grid really....
49882     },
49883     
49884     // private
49885     refreshValue: function() {
49886          var val = [];
49887         this.grid.getDataSource().each(function(r) {
49888             val.push(r.data);
49889         });
49890         this.el.dom.value = Roo.encode(val);
49891     }
49892     
49893      
49894     
49895     
49896 });/*
49897  * Based on:
49898  * Ext JS Library 1.1.1
49899  * Copyright(c) 2006-2007, Ext JS, LLC.
49900  *
49901  * Originally Released Under LGPL - original licence link has changed is not relivant.
49902  *
49903  * Fork - LGPL
49904  * <script type="text/javascript">
49905  */
49906 /**
49907  * @class Roo.form.DisplayField
49908  * @extends Roo.form.Field
49909  * A generic Field to display non-editable data.
49910  * @cfg {Boolean} closable (true|false) default false
49911  * @constructor
49912  * Creates a new Display Field item.
49913  * @param {Object} config Configuration options
49914  */
49915 Roo.form.DisplayField = function(config){
49916     Roo.form.DisplayField.superclass.constructor.call(this, config);
49917     
49918     this.addEvents({
49919         /**
49920          * @event close
49921          * Fires after the click the close btn
49922              * @param {Roo.form.DisplayField} this
49923              */
49924         close : true
49925     });
49926 };
49927
49928 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49929     inputType:      'hidden',
49930     allowBlank:     true,
49931     readOnly:         true,
49932     
49933  
49934     /**
49935      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49936      */
49937     focusClass : undefined,
49938     /**
49939      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49940      */
49941     fieldClass: 'x-form-field',
49942     
49943      /**
49944      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49945      */
49946     valueRenderer: undefined,
49947     
49948     width: 100,
49949     /**
49950      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49951      * {tag: "input", type: "checkbox", autocomplete: "off"})
49952      */
49953      
49954  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49955  
49956     closable : false,
49957     
49958     onResize : function(){
49959         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49960         
49961     },
49962
49963     initEvents : function(){
49964         // Roo.form.Checkbox.superclass.initEvents.call(this);
49965         // has no events...
49966         
49967         if(this.closable){
49968             this.closeEl.on('click', this.onClose, this);
49969         }
49970        
49971     },
49972
49973
49974     getResizeEl : function(){
49975         return this.wrap;
49976     },
49977
49978     getPositionEl : function(){
49979         return this.wrap;
49980     },
49981
49982     // private
49983     onRender : function(ct, position){
49984         
49985         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49986         //if(this.inputValue !== undefined){
49987         this.wrap = this.el.wrap();
49988         
49989         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49990         
49991         if(this.closable){
49992             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49993         }
49994         
49995         if (this.bodyStyle) {
49996             this.viewEl.applyStyles(this.bodyStyle);
49997         }
49998         //this.viewEl.setStyle('padding', '2px');
49999         
50000         this.setValue(this.value);
50001         
50002     },
50003 /*
50004     // private
50005     initValue : Roo.emptyFn,
50006
50007   */
50008
50009         // private
50010     onClick : function(){
50011         
50012     },
50013
50014     /**
50015      * Sets the checked state of the checkbox.
50016      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
50017      */
50018     setValue : function(v){
50019         this.value = v;
50020         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
50021         // this might be called before we have a dom element..
50022         if (!this.viewEl) {
50023             return;
50024         }
50025         this.viewEl.dom.innerHTML = html;
50026         Roo.form.DisplayField.superclass.setValue.call(this, v);
50027
50028     },
50029     
50030     onClose : function(e)
50031     {
50032         e.preventDefault();
50033         
50034         this.fireEvent('close', this);
50035     }
50036 });/*
50037  * 
50038  * Licence- LGPL
50039  * 
50040  */
50041
50042 /**
50043  * @class Roo.form.DayPicker
50044  * @extends Roo.form.Field
50045  * A Day picker show [M] [T] [W] ....
50046  * @constructor
50047  * Creates a new Day Picker
50048  * @param {Object} config Configuration options
50049  */
50050 Roo.form.DayPicker= function(config){
50051     Roo.form.DayPicker.superclass.constructor.call(this, config);
50052      
50053 };
50054
50055 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
50056     /**
50057      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
50058      */
50059     focusClass : undefined,
50060     /**
50061      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
50062      */
50063     fieldClass: "x-form-field",
50064    
50065     /**
50066      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
50067      * {tag: "input", type: "checkbox", autocomplete: "off"})
50068      */
50069     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
50070     
50071    
50072     actionMode : 'viewEl', 
50073     //
50074     // private
50075  
50076     inputType : 'hidden',
50077     
50078      
50079     inputElement: false, // real input element?
50080     basedOn: false, // ????
50081     
50082     isFormField: true, // not sure where this is needed!!!!
50083
50084     onResize : function(){
50085         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
50086         if(!this.boxLabel){
50087             this.el.alignTo(this.wrap, 'c-c');
50088         }
50089     },
50090
50091     initEvents : function(){
50092         Roo.form.Checkbox.superclass.initEvents.call(this);
50093         this.el.on("click", this.onClick,  this);
50094         this.el.on("change", this.onClick,  this);
50095     },
50096
50097
50098     getResizeEl : function(){
50099         return this.wrap;
50100     },
50101
50102     getPositionEl : function(){
50103         return this.wrap;
50104     },
50105
50106     
50107     // private
50108     onRender : function(ct, position){
50109         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
50110        
50111         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
50112         
50113         var r1 = '<table><tr>';
50114         var r2 = '<tr class="x-form-daypick-icons">';
50115         for (var i=0; i < 7; i++) {
50116             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
50117             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
50118         }
50119         
50120         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
50121         viewEl.select('img').on('click', this.onClick, this);
50122         this.viewEl = viewEl;   
50123         
50124         
50125         // this will not work on Chrome!!!
50126         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
50127         this.el.on('propertychange', this.setFromHidden,  this);  //ie
50128         
50129         
50130           
50131
50132     },
50133
50134     // private
50135     initValue : Roo.emptyFn,
50136
50137     /**
50138      * Returns the checked state of the checkbox.
50139      * @return {Boolean} True if checked, else false
50140      */
50141     getValue : function(){
50142         return this.el.dom.value;
50143         
50144     },
50145
50146         // private
50147     onClick : function(e){ 
50148         //this.setChecked(!this.checked);
50149         Roo.get(e.target).toggleClass('x-menu-item-checked');
50150         this.refreshValue();
50151         //if(this.el.dom.checked != this.checked){
50152         //    this.setValue(this.el.dom.checked);
50153        // }
50154     },
50155     
50156     // private
50157     refreshValue : function()
50158     {
50159         var val = '';
50160         this.viewEl.select('img',true).each(function(e,i,n)  {
50161             val += e.is(".x-menu-item-checked") ? String(n) : '';
50162         });
50163         this.setValue(val, true);
50164     },
50165
50166     /**
50167      * Sets the checked state of the checkbox.
50168      * On is always based on a string comparison between inputValue and the param.
50169      * @param {Boolean/String} value - the value to set 
50170      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50171      */
50172     setValue : function(v,suppressEvent){
50173         if (!this.el.dom) {
50174             return;
50175         }
50176         var old = this.el.dom.value ;
50177         this.el.dom.value = v;
50178         if (suppressEvent) {
50179             return ;
50180         }
50181          
50182         // update display..
50183         this.viewEl.select('img',true).each(function(e,i,n)  {
50184             
50185             var on = e.is(".x-menu-item-checked");
50186             var newv = v.indexOf(String(n)) > -1;
50187             if (on != newv) {
50188                 e.toggleClass('x-menu-item-checked');
50189             }
50190             
50191         });
50192         
50193         
50194         this.fireEvent('change', this, v, old);
50195         
50196         
50197     },
50198    
50199     // handle setting of hidden value by some other method!!?!?
50200     setFromHidden: function()
50201     {
50202         if(!this.el){
50203             return;
50204         }
50205         //console.log("SET FROM HIDDEN");
50206         //alert('setFrom hidden');
50207         this.setValue(this.el.dom.value);
50208     },
50209     
50210     onDestroy : function()
50211     {
50212         if(this.viewEl){
50213             Roo.get(this.viewEl).remove();
50214         }
50215          
50216         Roo.form.DayPicker.superclass.onDestroy.call(this);
50217     }
50218
50219 });/*
50220  * RooJS Library 1.1.1
50221  * Copyright(c) 2008-2011  Alan Knowles
50222  *
50223  * License - LGPL
50224  */
50225  
50226
50227 /**
50228  * @class Roo.form.ComboCheck
50229  * @extends Roo.form.ComboBox
50230  * A combobox for multiple select items.
50231  *
50232  * FIXME - could do with a reset button..
50233  * 
50234  * @constructor
50235  * Create a new ComboCheck
50236  * @param {Object} config Configuration options
50237  */
50238 Roo.form.ComboCheck = function(config){
50239     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50240     // should verify some data...
50241     // like
50242     // hiddenName = required..
50243     // displayField = required
50244     // valudField == required
50245     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50246     var _t = this;
50247     Roo.each(req, function(e) {
50248         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50249             throw "Roo.form.ComboCheck : missing value for: " + e;
50250         }
50251     });
50252     
50253     
50254 };
50255
50256 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50257      
50258      
50259     editable : false,
50260      
50261     selectedClass: 'x-menu-item-checked', 
50262     
50263     // private
50264     onRender : function(ct, position){
50265         var _t = this;
50266         
50267         
50268         
50269         if(!this.tpl){
50270             var cls = 'x-combo-list';
50271
50272             
50273             this.tpl =  new Roo.Template({
50274                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50275                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50276                    '<span>{' + this.displayField + '}</span>' +
50277                     '</div>' 
50278                 
50279             });
50280         }
50281  
50282         
50283         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50284         this.view.singleSelect = false;
50285         this.view.multiSelect = true;
50286         this.view.toggleSelect = true;
50287         this.pageTb.add(new Roo.Toolbar.Fill(), {
50288             
50289             text: 'Done',
50290             handler: function()
50291             {
50292                 _t.collapse();
50293             }
50294         });
50295     },
50296     
50297     onViewOver : function(e, t){
50298         // do nothing...
50299         return;
50300         
50301     },
50302     
50303     onViewClick : function(doFocus,index){
50304         return;
50305         
50306     },
50307     select: function () {
50308         //Roo.log("SELECT CALLED");
50309     },
50310      
50311     selectByValue : function(xv, scrollIntoView){
50312         var ar = this.getValueArray();
50313         var sels = [];
50314         
50315         Roo.each(ar, function(v) {
50316             if(v === undefined || v === null){
50317                 return;
50318             }
50319             var r = this.findRecord(this.valueField, v);
50320             if(r){
50321                 sels.push(this.store.indexOf(r))
50322                 
50323             }
50324         },this);
50325         this.view.select(sels);
50326         return false;
50327     },
50328     
50329     
50330     
50331     onSelect : function(record, index){
50332        // Roo.log("onselect Called");
50333        // this is only called by the clear button now..
50334         this.view.clearSelections();
50335         this.setValue('[]');
50336         if (this.value != this.valueBefore) {
50337             this.fireEvent('change', this, this.value, this.valueBefore);
50338             this.valueBefore = this.value;
50339         }
50340     },
50341     getValueArray : function()
50342     {
50343         var ar = [] ;
50344         
50345         try {
50346             //Roo.log(this.value);
50347             if (typeof(this.value) == 'undefined') {
50348                 return [];
50349             }
50350             var ar = Roo.decode(this.value);
50351             return  ar instanceof Array ? ar : []; //?? valid?
50352             
50353         } catch(e) {
50354             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50355             return [];
50356         }
50357          
50358     },
50359     expand : function ()
50360     {
50361         
50362         Roo.form.ComboCheck.superclass.expand.call(this);
50363         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50364         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50365         
50366
50367     },
50368     
50369     collapse : function(){
50370         Roo.form.ComboCheck.superclass.collapse.call(this);
50371         var sl = this.view.getSelectedIndexes();
50372         var st = this.store;
50373         var nv = [];
50374         var tv = [];
50375         var r;
50376         Roo.each(sl, function(i) {
50377             r = st.getAt(i);
50378             nv.push(r.get(this.valueField));
50379         },this);
50380         this.setValue(Roo.encode(nv));
50381         if (this.value != this.valueBefore) {
50382
50383             this.fireEvent('change', this, this.value, this.valueBefore);
50384             this.valueBefore = this.value;
50385         }
50386         
50387     },
50388     
50389     setValue : function(v){
50390         // Roo.log(v);
50391         this.value = v;
50392         
50393         var vals = this.getValueArray();
50394         var tv = [];
50395         Roo.each(vals, function(k) {
50396             var r = this.findRecord(this.valueField, k);
50397             if(r){
50398                 tv.push(r.data[this.displayField]);
50399             }else if(this.valueNotFoundText !== undefined){
50400                 tv.push( this.valueNotFoundText );
50401             }
50402         },this);
50403        // Roo.log(tv);
50404         
50405         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50406         this.hiddenField.value = v;
50407         this.value = v;
50408     }
50409     
50410 });/*
50411  * Based on:
50412  * Ext JS Library 1.1.1
50413  * Copyright(c) 2006-2007, Ext JS, LLC.
50414  *
50415  * Originally Released Under LGPL - original licence link has changed is not relivant.
50416  *
50417  * Fork - LGPL
50418  * <script type="text/javascript">
50419  */
50420  
50421 /**
50422  * @class Roo.form.Signature
50423  * @extends Roo.form.Field
50424  * Signature field.  
50425  * @constructor
50426  * 
50427  * @param {Object} config Configuration options
50428  */
50429
50430 Roo.form.Signature = function(config){
50431     Roo.form.Signature.superclass.constructor.call(this, config);
50432     
50433     this.addEvents({// not in used??
50434          /**
50435          * @event confirm
50436          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50437              * @param {Roo.form.Signature} combo This combo box
50438              */
50439         'confirm' : true,
50440         /**
50441          * @event reset
50442          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50443              * @param {Roo.form.ComboBox} combo This combo box
50444              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50445              */
50446         'reset' : true
50447     });
50448 };
50449
50450 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50451     /**
50452      * @cfg {Object} labels Label to use when rendering a form.
50453      * defaults to 
50454      * labels : { 
50455      *      clear : "Clear",
50456      *      confirm : "Confirm"
50457      *  }
50458      */
50459     labels : { 
50460         clear : "Clear",
50461         confirm : "Confirm"
50462     },
50463     /**
50464      * @cfg {Number} width The signature panel width (defaults to 300)
50465      */
50466     width: 300,
50467     /**
50468      * @cfg {Number} height The signature panel height (defaults to 100)
50469      */
50470     height : 100,
50471     /**
50472      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50473      */
50474     allowBlank : false,
50475     
50476     //private
50477     // {Object} signPanel The signature SVG panel element (defaults to {})
50478     signPanel : {},
50479     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50480     isMouseDown : false,
50481     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50482     isConfirmed : false,
50483     // {String} signatureTmp SVG mapping string (defaults to empty string)
50484     signatureTmp : '',
50485     
50486     
50487     defaultAutoCreate : { // modified by initCompnoent..
50488         tag: "input",
50489         type:"hidden"
50490     },
50491
50492     // private
50493     onRender : function(ct, position){
50494         
50495         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50496         
50497         this.wrap = this.el.wrap({
50498             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50499         });
50500         
50501         this.createToolbar(this);
50502         this.signPanel = this.wrap.createChild({
50503                 tag: 'div',
50504                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50505             }, this.el
50506         );
50507             
50508         this.svgID = Roo.id();
50509         this.svgEl = this.signPanel.createChild({
50510               xmlns : 'http://www.w3.org/2000/svg',
50511               tag : 'svg',
50512               id : this.svgID + "-svg",
50513               width: this.width,
50514               height: this.height,
50515               viewBox: '0 0 '+this.width+' '+this.height,
50516               cn : [
50517                 {
50518                     tag: "rect",
50519                     id: this.svgID + "-svg-r",
50520                     width: this.width,
50521                     height: this.height,
50522                     fill: "#ffa"
50523                 },
50524                 {
50525                     tag: "line",
50526                     id: this.svgID + "-svg-l",
50527                     x1: "0", // start
50528                     y1: (this.height*0.8), // start set the line in 80% of height
50529                     x2: this.width, // end
50530                     y2: (this.height*0.8), // end set the line in 80% of height
50531                     'stroke': "#666",
50532                     'stroke-width': "1",
50533                     'stroke-dasharray': "3",
50534                     'shape-rendering': "crispEdges",
50535                     'pointer-events': "none"
50536                 },
50537                 {
50538                     tag: "path",
50539                     id: this.svgID + "-svg-p",
50540                     'stroke': "navy",
50541                     'stroke-width': "3",
50542                     'fill': "none",
50543                     'pointer-events': 'none'
50544                 }
50545               ]
50546         });
50547         this.createSVG();
50548         this.svgBox = this.svgEl.dom.getScreenCTM();
50549     },
50550     createSVG : function(){ 
50551         var svg = this.signPanel;
50552         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50553         var t = this;
50554
50555         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50556         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50557         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50558         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50559         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50560         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50561         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50562         
50563     },
50564     isTouchEvent : function(e){
50565         return e.type.match(/^touch/);
50566     },
50567     getCoords : function (e) {
50568         var pt    = this.svgEl.dom.createSVGPoint();
50569         pt.x = e.clientX; 
50570         pt.y = e.clientY;
50571         if (this.isTouchEvent(e)) {
50572             pt.x =  e.targetTouches[0].clientX;
50573             pt.y = e.targetTouches[0].clientY;
50574         }
50575         var a = this.svgEl.dom.getScreenCTM();
50576         var b = a.inverse();
50577         var mx = pt.matrixTransform(b);
50578         return mx.x + ',' + mx.y;
50579     },
50580     //mouse event headler 
50581     down : function (e) {
50582         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50583         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50584         
50585         this.isMouseDown = true;
50586         
50587         e.preventDefault();
50588     },
50589     move : function (e) {
50590         if (this.isMouseDown) {
50591             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50592             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50593         }
50594         
50595         e.preventDefault();
50596     },
50597     up : function (e) {
50598         this.isMouseDown = false;
50599         var sp = this.signatureTmp.split(' ');
50600         
50601         if(sp.length > 1){
50602             if(!sp[sp.length-2].match(/^L/)){
50603                 sp.pop();
50604                 sp.pop();
50605                 sp.push("");
50606                 this.signatureTmp = sp.join(" ");
50607             }
50608         }
50609         if(this.getValue() != this.signatureTmp){
50610             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50611             this.isConfirmed = false;
50612         }
50613         e.preventDefault();
50614     },
50615     
50616     /**
50617      * Protected method that will not generally be called directly. It
50618      * is called when the editor creates its toolbar. Override this method if you need to
50619      * add custom toolbar buttons.
50620      * @param {HtmlEditor} editor
50621      */
50622     createToolbar : function(editor){
50623          function btn(id, toggle, handler){
50624             var xid = fid + '-'+ id ;
50625             return {
50626                 id : xid,
50627                 cmd : id,
50628                 cls : 'x-btn-icon x-edit-'+id,
50629                 enableToggle:toggle !== false,
50630                 scope: editor, // was editor...
50631                 handler:handler||editor.relayBtnCmd,
50632                 clickEvent:'mousedown',
50633                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50634                 tabIndex:-1
50635             };
50636         }
50637         
50638         
50639         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50640         this.tb = tb;
50641         this.tb.add(
50642            {
50643                 cls : ' x-signature-btn x-signature-'+id,
50644                 scope: editor, // was editor...
50645                 handler: this.reset,
50646                 clickEvent:'mousedown',
50647                 text: this.labels.clear
50648             },
50649             {
50650                  xtype : 'Fill',
50651                  xns: Roo.Toolbar
50652             }, 
50653             {
50654                 cls : '  x-signature-btn x-signature-'+id,
50655                 scope: editor, // was editor...
50656                 handler: this.confirmHandler,
50657                 clickEvent:'mousedown',
50658                 text: this.labels.confirm
50659             }
50660         );
50661     
50662     },
50663     //public
50664     /**
50665      * when user is clicked confirm then show this image.....
50666      * 
50667      * @return {String} Image Data URI
50668      */
50669     getImageDataURI : function(){
50670         var svg = this.svgEl.dom.parentNode.innerHTML;
50671         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50672         return src; 
50673     },
50674     /**
50675      * 
50676      * @return {Boolean} this.isConfirmed
50677      */
50678     getConfirmed : function(){
50679         return this.isConfirmed;
50680     },
50681     /**
50682      * 
50683      * @return {Number} this.width
50684      */
50685     getWidth : function(){
50686         return this.width;
50687     },
50688     /**
50689      * 
50690      * @return {Number} this.height
50691      */
50692     getHeight : function(){
50693         return this.height;
50694     },
50695     // private
50696     getSignature : function(){
50697         return this.signatureTmp;
50698     },
50699     // private
50700     reset : function(){
50701         this.signatureTmp = '';
50702         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50703         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50704         this.isConfirmed = false;
50705         Roo.form.Signature.superclass.reset.call(this);
50706     },
50707     setSignature : function(s){
50708         this.signatureTmp = s;
50709         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50710         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50711         this.setValue(s);
50712         this.isConfirmed = false;
50713         Roo.form.Signature.superclass.reset.call(this);
50714     }, 
50715     test : function(){
50716 //        Roo.log(this.signPanel.dom.contentWindow.up())
50717     },
50718     //private
50719     setConfirmed : function(){
50720         
50721         
50722         
50723 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50724     },
50725     // private
50726     confirmHandler : function(){
50727         if(!this.getSignature()){
50728             return;
50729         }
50730         
50731         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50732         this.setValue(this.getSignature());
50733         this.isConfirmed = true;
50734         
50735         this.fireEvent('confirm', this);
50736     },
50737     // private
50738     // Subclasses should provide the validation implementation by overriding this
50739     validateValue : function(value){
50740         if(this.allowBlank){
50741             return true;
50742         }
50743         
50744         if(this.isConfirmed){
50745             return true;
50746         }
50747         return false;
50748     }
50749 });/*
50750  * Based on:
50751  * Ext JS Library 1.1.1
50752  * Copyright(c) 2006-2007, Ext JS, LLC.
50753  *
50754  * Originally Released Under LGPL - original licence link has changed is not relivant.
50755  *
50756  * Fork - LGPL
50757  * <script type="text/javascript">
50758  */
50759  
50760
50761 /**
50762  * @class Roo.form.ComboBox
50763  * @extends Roo.form.TriggerField
50764  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50765  * @constructor
50766  * Create a new ComboBox.
50767  * @param {Object} config Configuration options
50768  */
50769 Roo.form.Select = function(config){
50770     Roo.form.Select.superclass.constructor.call(this, config);
50771      
50772 };
50773
50774 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50775     /**
50776      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50777      */
50778     /**
50779      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50780      * rendering into an Roo.Editor, defaults to false)
50781      */
50782     /**
50783      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50784      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50785      */
50786     /**
50787      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50788      */
50789     /**
50790      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50791      * the dropdown list (defaults to undefined, with no header element)
50792      */
50793
50794      /**
50795      * @cfg {String/Roo.Template} tpl The template to use to render the output
50796      */
50797      
50798     // private
50799     defaultAutoCreate : {tag: "select"  },
50800     /**
50801      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50802      */
50803     listWidth: undefined,
50804     /**
50805      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50806      * mode = 'remote' or 'text' if mode = 'local')
50807      */
50808     displayField: undefined,
50809     /**
50810      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50811      * mode = 'remote' or 'value' if mode = 'local'). 
50812      * Note: use of a valueField requires the user make a selection
50813      * in order for a value to be mapped.
50814      */
50815     valueField: undefined,
50816     
50817     
50818     /**
50819      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50820      * field's data value (defaults to the underlying DOM element's name)
50821      */
50822     hiddenName: undefined,
50823     /**
50824      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50825      */
50826     listClass: '',
50827     /**
50828      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50829      */
50830     selectedClass: 'x-combo-selected',
50831     /**
50832      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50833      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50834      * which displays a downward arrow icon).
50835      */
50836     triggerClass : 'x-form-arrow-trigger',
50837     /**
50838      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50839      */
50840     shadow:'sides',
50841     /**
50842      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50843      * anchor positions (defaults to 'tl-bl')
50844      */
50845     listAlign: 'tl-bl?',
50846     /**
50847      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50848      */
50849     maxHeight: 300,
50850     /**
50851      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50852      * query specified by the allQuery config option (defaults to 'query')
50853      */
50854     triggerAction: 'query',
50855     /**
50856      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50857      * (defaults to 4, does not apply if editable = false)
50858      */
50859     minChars : 4,
50860     /**
50861      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50862      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50863      */
50864     typeAhead: false,
50865     /**
50866      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50867      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50868      */
50869     queryDelay: 500,
50870     /**
50871      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50872      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50873      */
50874     pageSize: 0,
50875     /**
50876      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50877      * when editable = true (defaults to false)
50878      */
50879     selectOnFocus:false,
50880     /**
50881      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50882      */
50883     queryParam: 'query',
50884     /**
50885      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50886      * when mode = 'remote' (defaults to 'Loading...')
50887      */
50888     loadingText: 'Loading...',
50889     /**
50890      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50891      */
50892     resizable: false,
50893     /**
50894      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50895      */
50896     handleHeight : 8,
50897     /**
50898      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50899      * traditional select (defaults to true)
50900      */
50901     editable: true,
50902     /**
50903      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50904      */
50905     allQuery: '',
50906     /**
50907      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50908      */
50909     mode: 'remote',
50910     /**
50911      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50912      * listWidth has a higher value)
50913      */
50914     minListWidth : 70,
50915     /**
50916      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50917      * allow the user to set arbitrary text into the field (defaults to false)
50918      */
50919     forceSelection:false,
50920     /**
50921      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50922      * if typeAhead = true (defaults to 250)
50923      */
50924     typeAheadDelay : 250,
50925     /**
50926      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50927      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50928      */
50929     valueNotFoundText : undefined,
50930     
50931     /**
50932      * @cfg {String} defaultValue The value displayed after loading the store.
50933      */
50934     defaultValue: '',
50935     
50936     /**
50937      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50938      */
50939     blockFocus : false,
50940     
50941     /**
50942      * @cfg {Boolean} disableClear Disable showing of clear button.
50943      */
50944     disableClear : false,
50945     /**
50946      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50947      */
50948     alwaysQuery : false,
50949     
50950     //private
50951     addicon : false,
50952     editicon: false,
50953     
50954     // element that contains real text value.. (when hidden is used..)
50955      
50956     // private
50957     onRender : function(ct, position){
50958         Roo.form.Field.prototype.onRender.call(this, ct, position);
50959         
50960         if(this.store){
50961             this.store.on('beforeload', this.onBeforeLoad, this);
50962             this.store.on('load', this.onLoad, this);
50963             this.store.on('loadexception', this.onLoadException, this);
50964             this.store.load({});
50965         }
50966         
50967         
50968         
50969     },
50970
50971     // private
50972     initEvents : function(){
50973         //Roo.form.ComboBox.superclass.initEvents.call(this);
50974  
50975     },
50976
50977     onDestroy : function(){
50978        
50979         if(this.store){
50980             this.store.un('beforeload', this.onBeforeLoad, this);
50981             this.store.un('load', this.onLoad, this);
50982             this.store.un('loadexception', this.onLoadException, this);
50983         }
50984         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50985     },
50986
50987     // private
50988     fireKey : function(e){
50989         if(e.isNavKeyPress() && !this.list.isVisible()){
50990             this.fireEvent("specialkey", this, e);
50991         }
50992     },
50993
50994     // private
50995     onResize: function(w, h){
50996         
50997         return; 
50998     
50999         
51000     },
51001
51002     /**
51003      * Allow or prevent the user from directly editing the field text.  If false is passed,
51004      * the user will only be able to select from the items defined in the dropdown list.  This method
51005      * is the runtime equivalent of setting the 'editable' config option at config time.
51006      * @param {Boolean} value True to allow the user to directly edit the field text
51007      */
51008     setEditable : function(value){
51009          
51010     },
51011
51012     // private
51013     onBeforeLoad : function(){
51014         
51015         Roo.log("Select before load");
51016         return;
51017     
51018         this.innerList.update(this.loadingText ?
51019                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
51020         //this.restrictHeight();
51021         this.selectedIndex = -1;
51022     },
51023
51024     // private
51025     onLoad : function(){
51026
51027     
51028         var dom = this.el.dom;
51029         dom.innerHTML = '';
51030          var od = dom.ownerDocument;
51031          
51032         if (this.emptyText) {
51033             var op = od.createElement('option');
51034             op.setAttribute('value', '');
51035             op.innerHTML = String.format('{0}', this.emptyText);
51036             dom.appendChild(op);
51037         }
51038         if(this.store.getCount() > 0){
51039            
51040             var vf = this.valueField;
51041             var df = this.displayField;
51042             this.store.data.each(function(r) {
51043                 // which colmsn to use... testing - cdoe / title..
51044                 var op = od.createElement('option');
51045                 op.setAttribute('value', r.data[vf]);
51046                 op.innerHTML = String.format('{0}', r.data[df]);
51047                 dom.appendChild(op);
51048             });
51049             if (typeof(this.defaultValue != 'undefined')) {
51050                 this.setValue(this.defaultValue);
51051             }
51052             
51053              
51054         }else{
51055             //this.onEmptyResults();
51056         }
51057         //this.el.focus();
51058     },
51059     // private
51060     onLoadException : function()
51061     {
51062         dom.innerHTML = '';
51063             
51064         Roo.log("Select on load exception");
51065         return;
51066     
51067         this.collapse();
51068         Roo.log(this.store.reader.jsonData);
51069         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
51070             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
51071         }
51072         
51073         
51074     },
51075     // private
51076     onTypeAhead : function(){
51077          
51078     },
51079
51080     // private
51081     onSelect : function(record, index){
51082         Roo.log('on select?');
51083         return;
51084         if(this.fireEvent('beforeselect', this, record, index) !== false){
51085             this.setFromData(index > -1 ? record.data : false);
51086             this.collapse();
51087             this.fireEvent('select', this, record, index);
51088         }
51089     },
51090
51091     /**
51092      * Returns the currently selected field value or empty string if no value is set.
51093      * @return {String} value The selected value
51094      */
51095     getValue : function(){
51096         var dom = this.el.dom;
51097         this.value = dom.options[dom.selectedIndex].value;
51098         return this.value;
51099         
51100     },
51101
51102     /**
51103      * Clears any text/value currently set in the field
51104      */
51105     clearValue : function(){
51106         this.value = '';
51107         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
51108         
51109     },
51110
51111     /**
51112      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
51113      * will be displayed in the field.  If the value does not match the data value of an existing item,
51114      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
51115      * Otherwise the field will be blank (although the value will still be set).
51116      * @param {String} value The value to match
51117      */
51118     setValue : function(v){
51119         var d = this.el.dom;
51120         for (var i =0; i < d.options.length;i++) {
51121             if (v == d.options[i].value) {
51122                 d.selectedIndex = i;
51123                 this.value = v;
51124                 return;
51125             }
51126         }
51127         this.clearValue();
51128     },
51129     /**
51130      * @property {Object} the last set data for the element
51131      */
51132     
51133     lastData : false,
51134     /**
51135      * Sets the value of the field based on a object which is related to the record format for the store.
51136      * @param {Object} value the value to set as. or false on reset?
51137      */
51138     setFromData : function(o){
51139         Roo.log('setfrom data?');
51140          
51141         
51142         
51143     },
51144     // private
51145     reset : function(){
51146         this.clearValue();
51147     },
51148     // private
51149     findRecord : function(prop, value){
51150         
51151         return false;
51152     
51153         var record;
51154         if(this.store.getCount() > 0){
51155             this.store.each(function(r){
51156                 if(r.data[prop] == value){
51157                     record = r;
51158                     return false;
51159                 }
51160                 return true;
51161             });
51162         }
51163         return record;
51164     },
51165     
51166     getName: function()
51167     {
51168         // returns hidden if it's set..
51169         if (!this.rendered) {return ''};
51170         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51171         
51172     },
51173      
51174
51175     
51176
51177     // private
51178     onEmptyResults : function(){
51179         Roo.log('empty results');
51180         //this.collapse();
51181     },
51182
51183     /**
51184      * Returns true if the dropdown list is expanded, else false.
51185      */
51186     isExpanded : function(){
51187         return false;
51188     },
51189
51190     /**
51191      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51192      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51193      * @param {String} value The data value of the item to select
51194      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51195      * selected item if it is not currently in view (defaults to true)
51196      * @return {Boolean} True if the value matched an item in the list, else false
51197      */
51198     selectByValue : function(v, scrollIntoView){
51199         Roo.log('select By Value');
51200         return false;
51201     
51202         if(v !== undefined && v !== null){
51203             var r = this.findRecord(this.valueField || this.displayField, v);
51204             if(r){
51205                 this.select(this.store.indexOf(r), scrollIntoView);
51206                 return true;
51207             }
51208         }
51209         return false;
51210     },
51211
51212     /**
51213      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51214      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51215      * @param {Number} index The zero-based index of the list item to select
51216      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51217      * selected item if it is not currently in view (defaults to true)
51218      */
51219     select : function(index, scrollIntoView){
51220         Roo.log('select ');
51221         return  ;
51222         
51223         this.selectedIndex = index;
51224         this.view.select(index);
51225         if(scrollIntoView !== false){
51226             var el = this.view.getNode(index);
51227             if(el){
51228                 this.innerList.scrollChildIntoView(el, false);
51229             }
51230         }
51231     },
51232
51233       
51234
51235     // private
51236     validateBlur : function(){
51237         
51238         return;
51239         
51240     },
51241
51242     // private
51243     initQuery : function(){
51244         this.doQuery(this.getRawValue());
51245     },
51246
51247     // private
51248     doForce : function(){
51249         if(this.el.dom.value.length > 0){
51250             this.el.dom.value =
51251                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51252              
51253         }
51254     },
51255
51256     /**
51257      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51258      * query allowing the query action to be canceled if needed.
51259      * @param {String} query The SQL query to execute
51260      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51261      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51262      * saved in the current store (defaults to false)
51263      */
51264     doQuery : function(q, forceAll){
51265         
51266         Roo.log('doQuery?');
51267         if(q === undefined || q === null){
51268             q = '';
51269         }
51270         var qe = {
51271             query: q,
51272             forceAll: forceAll,
51273             combo: this,
51274             cancel:false
51275         };
51276         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51277             return false;
51278         }
51279         q = qe.query;
51280         forceAll = qe.forceAll;
51281         if(forceAll === true || (q.length >= this.minChars)){
51282             if(this.lastQuery != q || this.alwaysQuery){
51283                 this.lastQuery = q;
51284                 if(this.mode == 'local'){
51285                     this.selectedIndex = -1;
51286                     if(forceAll){
51287                         this.store.clearFilter();
51288                     }else{
51289                         this.store.filter(this.displayField, q);
51290                     }
51291                     this.onLoad();
51292                 }else{
51293                     this.store.baseParams[this.queryParam] = q;
51294                     this.store.load({
51295                         params: this.getParams(q)
51296                     });
51297                     this.expand();
51298                 }
51299             }else{
51300                 this.selectedIndex = -1;
51301                 this.onLoad();   
51302             }
51303         }
51304     },
51305
51306     // private
51307     getParams : function(q){
51308         var p = {};
51309         //p[this.queryParam] = q;
51310         if(this.pageSize){
51311             p.start = 0;
51312             p.limit = this.pageSize;
51313         }
51314         return p;
51315     },
51316
51317     /**
51318      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51319      */
51320     collapse : function(){
51321         
51322     },
51323
51324     // private
51325     collapseIf : function(e){
51326         
51327     },
51328
51329     /**
51330      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51331      */
51332     expand : function(){
51333         
51334     } ,
51335
51336     // private
51337      
51338
51339     /** 
51340     * @cfg {Boolean} grow 
51341     * @hide 
51342     */
51343     /** 
51344     * @cfg {Number} growMin 
51345     * @hide 
51346     */
51347     /** 
51348     * @cfg {Number} growMax 
51349     * @hide 
51350     */
51351     /**
51352      * @hide
51353      * @method autoSize
51354      */
51355     
51356     setWidth : function()
51357     {
51358         
51359     },
51360     getResizeEl : function(){
51361         return this.el;
51362     }
51363 });//<script type="text/javasscript">
51364  
51365
51366 /**
51367  * @class Roo.DDView
51368  * A DnD enabled version of Roo.View.
51369  * @param {Element/String} container The Element in which to create the View.
51370  * @param {String} tpl The template string used to create the markup for each element of the View
51371  * @param {Object} config The configuration properties. These include all the config options of
51372  * {@link Roo.View} plus some specific to this class.<br>
51373  * <p>
51374  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51375  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51376  * <p>
51377  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51378 .x-view-drag-insert-above {
51379         border-top:1px dotted #3366cc;
51380 }
51381 .x-view-drag-insert-below {
51382         border-bottom:1px dotted #3366cc;
51383 }
51384 </code></pre>
51385  * 
51386  */
51387  
51388 Roo.DDView = function(container, tpl, config) {
51389     Roo.DDView.superclass.constructor.apply(this, arguments);
51390     this.getEl().setStyle("outline", "0px none");
51391     this.getEl().unselectable();
51392     if (this.dragGroup) {
51393                 this.setDraggable(this.dragGroup.split(","));
51394     }
51395     if (this.dropGroup) {
51396                 this.setDroppable(this.dropGroup.split(","));
51397     }
51398     if (this.deletable) {
51399         this.setDeletable();
51400     }
51401     this.isDirtyFlag = false;
51402         this.addEvents({
51403                 "drop" : true
51404         });
51405 };
51406
51407 Roo.extend(Roo.DDView, Roo.View, {
51408 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51409 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51410 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51411 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51412
51413         isFormField: true,
51414
51415         reset: Roo.emptyFn,
51416         
51417         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51418
51419         validate: function() {
51420                 return true;
51421         },
51422         
51423         destroy: function() {
51424                 this.purgeListeners();
51425                 this.getEl.removeAllListeners();
51426                 this.getEl().remove();
51427                 if (this.dragZone) {
51428                         if (this.dragZone.destroy) {
51429                                 this.dragZone.destroy();
51430                         }
51431                 }
51432                 if (this.dropZone) {
51433                         if (this.dropZone.destroy) {
51434                                 this.dropZone.destroy();
51435                         }
51436                 }
51437         },
51438
51439 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51440         getName: function() {
51441                 return this.name;
51442         },
51443
51444 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51445         setValue: function(v) {
51446                 if (!this.store) {
51447                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51448                 }
51449                 var data = {};
51450                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51451                 this.store.proxy = new Roo.data.MemoryProxy(data);
51452                 this.store.load();
51453         },
51454
51455 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51456         getValue: function() {
51457                 var result = '(';
51458                 this.store.each(function(rec) {
51459                         result += rec.id + ',';
51460                 });
51461                 return result.substr(0, result.length - 1) + ')';
51462         },
51463         
51464         getIds: function() {
51465                 var i = 0, result = new Array(this.store.getCount());
51466                 this.store.each(function(rec) {
51467                         result[i++] = rec.id;
51468                 });
51469                 return result;
51470         },
51471         
51472         isDirty: function() {
51473                 return this.isDirtyFlag;
51474         },
51475
51476 /**
51477  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51478  *      whole Element becomes the target, and this causes the drop gesture to append.
51479  */
51480     getTargetFromEvent : function(e) {
51481                 var target = e.getTarget();
51482                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51483                 target = target.parentNode;
51484                 }
51485                 if (!target) {
51486                         target = this.el.dom.lastChild || this.el.dom;
51487                 }
51488                 return target;
51489     },
51490
51491 /**
51492  *      Create the drag data which consists of an object which has the property "ddel" as
51493  *      the drag proxy element. 
51494  */
51495     getDragData : function(e) {
51496         var target = this.findItemFromChild(e.getTarget());
51497                 if(target) {
51498                         this.handleSelection(e);
51499                         var selNodes = this.getSelectedNodes();
51500             var dragData = {
51501                 source: this,
51502                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51503                 nodes: selNodes,
51504                 records: []
51505                         };
51506                         var selectedIndices = this.getSelectedIndexes();
51507                         for (var i = 0; i < selectedIndices.length; i++) {
51508                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51509                         }
51510                         if (selNodes.length == 1) {
51511                                 dragData.ddel = target.cloneNode(true); // the div element
51512                         } else {
51513                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51514                                 div.className = 'multi-proxy';
51515                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51516                                         div.appendChild(selNodes[i].cloneNode(true));
51517                                 }
51518                                 dragData.ddel = div;
51519                         }
51520             //console.log(dragData)
51521             //console.log(dragData.ddel.innerHTML)
51522                         return dragData;
51523                 }
51524         //console.log('nodragData')
51525                 return false;
51526     },
51527     
51528 /**     Specify to which ddGroup items in this DDView may be dragged. */
51529     setDraggable: function(ddGroup) {
51530         if (ddGroup instanceof Array) {
51531                 Roo.each(ddGroup, this.setDraggable, this);
51532                 return;
51533         }
51534         if (this.dragZone) {
51535                 this.dragZone.addToGroup(ddGroup);
51536         } else {
51537                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51538                                 containerScroll: true,
51539                                 ddGroup: ddGroup 
51540
51541                         });
51542 //                      Draggability implies selection. DragZone's mousedown selects the element.
51543                         if (!this.multiSelect) { this.singleSelect = true; }
51544
51545 //                      Wire the DragZone's handlers up to methods in *this*
51546                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51547                 }
51548     },
51549
51550 /**     Specify from which ddGroup this DDView accepts drops. */
51551     setDroppable: function(ddGroup) {
51552         if (ddGroup instanceof Array) {
51553                 Roo.each(ddGroup, this.setDroppable, this);
51554                 return;
51555         }
51556         if (this.dropZone) {
51557                 this.dropZone.addToGroup(ddGroup);
51558         } else {
51559                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51560                                 containerScroll: true,
51561                                 ddGroup: ddGroup
51562                         });
51563
51564 //                      Wire the DropZone's handlers up to methods in *this*
51565                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51566                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51567                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51568                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51569                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51570                 }
51571     },
51572
51573 /**     Decide whether to drop above or below a View node. */
51574     getDropPoint : function(e, n, dd){
51575         if (n == this.el.dom) { return "above"; }
51576                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51577                 var c = t + (b - t) / 2;
51578                 var y = Roo.lib.Event.getPageY(e);
51579                 if(y <= c) {
51580                         return "above";
51581                 }else{
51582                         return "below";
51583                 }
51584     },
51585
51586     onNodeEnter : function(n, dd, e, data){
51587                 return false;
51588     },
51589     
51590     onNodeOver : function(n, dd, e, data){
51591                 var pt = this.getDropPoint(e, n, dd);
51592                 // set the insert point style on the target node
51593                 var dragElClass = this.dropNotAllowed;
51594                 if (pt) {
51595                         var targetElClass;
51596                         if (pt == "above"){
51597                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51598                                 targetElClass = "x-view-drag-insert-above";
51599                         } else {
51600                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51601                                 targetElClass = "x-view-drag-insert-below";
51602                         }
51603                         if (this.lastInsertClass != targetElClass){
51604                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51605                                 this.lastInsertClass = targetElClass;
51606                         }
51607                 }
51608                 return dragElClass;
51609         },
51610
51611     onNodeOut : function(n, dd, e, data){
51612                 this.removeDropIndicators(n);
51613     },
51614
51615     onNodeDrop : function(n, dd, e, data){
51616         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51617                 return false;
51618         }
51619         var pt = this.getDropPoint(e, n, dd);
51620                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51621                 if (pt == "below") { insertAt++; }
51622                 for (var i = 0; i < data.records.length; i++) {
51623                         var r = data.records[i];
51624                         var dup = this.store.getById(r.id);
51625                         if (dup && (dd != this.dragZone)) {
51626                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51627                         } else {
51628                                 if (data.copy) {
51629                                         this.store.insert(insertAt++, r.copy());
51630                                 } else {
51631                                         data.source.isDirtyFlag = true;
51632                                         r.store.remove(r);
51633                                         this.store.insert(insertAt++, r);
51634                                 }
51635                                 this.isDirtyFlag = true;
51636                         }
51637                 }
51638                 this.dragZone.cachedTarget = null;
51639                 return true;
51640     },
51641
51642     removeDropIndicators : function(n){
51643                 if(n){
51644                         Roo.fly(n).removeClass([
51645                                 "x-view-drag-insert-above",
51646                                 "x-view-drag-insert-below"]);
51647                         this.lastInsertClass = "_noclass";
51648                 }
51649     },
51650
51651 /**
51652  *      Utility method. Add a delete option to the DDView's context menu.
51653  *      @param {String} imageUrl The URL of the "delete" icon image.
51654  */
51655         setDeletable: function(imageUrl) {
51656                 if (!this.singleSelect && !this.multiSelect) {
51657                         this.singleSelect = true;
51658                 }
51659                 var c = this.getContextMenu();
51660                 this.contextMenu.on("itemclick", function(item) {
51661                         switch (item.id) {
51662                                 case "delete":
51663                                         this.remove(this.getSelectedIndexes());
51664                                         break;
51665                         }
51666                 }, this);
51667                 this.contextMenu.add({
51668                         icon: imageUrl,
51669                         id: "delete",
51670                         text: 'Delete'
51671                 });
51672         },
51673         
51674 /**     Return the context menu for this DDView. */
51675         getContextMenu: function() {
51676                 if (!this.contextMenu) {
51677 //                      Create the View's context menu
51678                         this.contextMenu = new Roo.menu.Menu({
51679                                 id: this.id + "-contextmenu"
51680                         });
51681                         this.el.on("contextmenu", this.showContextMenu, this);
51682                 }
51683                 return this.contextMenu;
51684         },
51685         
51686         disableContextMenu: function() {
51687                 if (this.contextMenu) {
51688                         this.el.un("contextmenu", this.showContextMenu, this);
51689                 }
51690         },
51691
51692         showContextMenu: function(e, item) {
51693         item = this.findItemFromChild(e.getTarget());
51694                 if (item) {
51695                         e.stopEvent();
51696                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51697                         this.contextMenu.showAt(e.getXY());
51698             }
51699     },
51700
51701 /**
51702  *      Remove {@link Roo.data.Record}s at the specified indices.
51703  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51704  */
51705     remove: function(selectedIndices) {
51706                 selectedIndices = [].concat(selectedIndices);
51707                 for (var i = 0; i < selectedIndices.length; i++) {
51708                         var rec = this.store.getAt(selectedIndices[i]);
51709                         this.store.remove(rec);
51710                 }
51711     },
51712
51713 /**
51714  *      Double click fires the event, but also, if this is draggable, and there is only one other
51715  *      related DropZone, it transfers the selected node.
51716  */
51717     onDblClick : function(e){
51718         var item = this.findItemFromChild(e.getTarget());
51719         if(item){
51720             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51721                 return false;
51722             }
51723             if (this.dragGroup) {
51724                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51725                     while (targets.indexOf(this.dropZone) > -1) {
51726                             targets.remove(this.dropZone);
51727                                 }
51728                     if (targets.length == 1) {
51729                                         this.dragZone.cachedTarget = null;
51730                         var el = Roo.get(targets[0].getEl());
51731                         var box = el.getBox(true);
51732                         targets[0].onNodeDrop(el.dom, {
51733                                 target: el.dom,
51734                                 xy: [box.x, box.y + box.height - 1]
51735                         }, null, this.getDragData(e));
51736                     }
51737                 }
51738         }
51739     },
51740     
51741     handleSelection: function(e) {
51742                 this.dragZone.cachedTarget = null;
51743         var item = this.findItemFromChild(e.getTarget());
51744         if (!item) {
51745                 this.clearSelections(true);
51746                 return;
51747         }
51748                 if (item && (this.multiSelect || this.singleSelect)){
51749                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51750                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51751                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51752                                 this.unselect(item);
51753                         } else {
51754                                 this.select(item, this.multiSelect && e.ctrlKey);
51755                                 this.lastSelection = item;
51756                         }
51757                 }
51758     },
51759
51760     onItemClick : function(item, index, e){
51761                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51762                         return false;
51763                 }
51764                 return true;
51765     },
51766
51767     unselect : function(nodeInfo, suppressEvent){
51768                 var node = this.getNode(nodeInfo);
51769                 if(node && this.isSelected(node)){
51770                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51771                                 Roo.fly(node).removeClass(this.selectedClass);
51772                                 this.selections.remove(node);
51773                                 if(!suppressEvent){
51774                                         this.fireEvent("selectionchange", this, this.selections);
51775                                 }
51776                         }
51777                 }
51778     }
51779 });
51780 /*
51781  * Based on:
51782  * Ext JS Library 1.1.1
51783  * Copyright(c) 2006-2007, Ext JS, LLC.
51784  *
51785  * Originally Released Under LGPL - original licence link has changed is not relivant.
51786  *
51787  * Fork - LGPL
51788  * <script type="text/javascript">
51789  */
51790  
51791 /**
51792  * @class Roo.LayoutManager
51793  * @extends Roo.util.Observable
51794  * Base class for layout managers.
51795  */
51796 Roo.LayoutManager = function(container, config){
51797     Roo.LayoutManager.superclass.constructor.call(this);
51798     this.el = Roo.get(container);
51799     // ie scrollbar fix
51800     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51801         document.body.scroll = "no";
51802     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51803         this.el.position('relative');
51804     }
51805     this.id = this.el.id;
51806     this.el.addClass("x-layout-container");
51807     /** false to disable window resize monitoring @type Boolean */
51808     this.monitorWindowResize = true;
51809     this.regions = {};
51810     this.addEvents({
51811         /**
51812          * @event layout
51813          * Fires when a layout is performed. 
51814          * @param {Roo.LayoutManager} this
51815          */
51816         "layout" : true,
51817         /**
51818          * @event regionresized
51819          * Fires when the user resizes a region. 
51820          * @param {Roo.LayoutRegion} region The resized region
51821          * @param {Number} newSize The new size (width for east/west, height for north/south)
51822          */
51823         "regionresized" : true,
51824         /**
51825          * @event regioncollapsed
51826          * Fires when a region is collapsed. 
51827          * @param {Roo.LayoutRegion} region The collapsed region
51828          */
51829         "regioncollapsed" : true,
51830         /**
51831          * @event regionexpanded
51832          * Fires when a region is expanded.  
51833          * @param {Roo.LayoutRegion} region The expanded region
51834          */
51835         "regionexpanded" : true
51836     });
51837     this.updating = false;
51838     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51839 };
51840
51841 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51842     /**
51843      * Returns true if this layout is currently being updated
51844      * @return {Boolean}
51845      */
51846     isUpdating : function(){
51847         return this.updating; 
51848     },
51849     
51850     /**
51851      * Suspend the LayoutManager from doing auto-layouts while
51852      * making multiple add or remove calls
51853      */
51854     beginUpdate : function(){
51855         this.updating = true;    
51856     },
51857     
51858     /**
51859      * Restore auto-layouts and optionally disable the manager from performing a layout
51860      * @param {Boolean} noLayout true to disable a layout update 
51861      */
51862     endUpdate : function(noLayout){
51863         this.updating = false;
51864         if(!noLayout){
51865             this.layout();
51866         }    
51867     },
51868     
51869     layout: function(){
51870         
51871     },
51872     
51873     onRegionResized : function(region, newSize){
51874         this.fireEvent("regionresized", region, newSize);
51875         this.layout();
51876     },
51877     
51878     onRegionCollapsed : function(region){
51879         this.fireEvent("regioncollapsed", region);
51880     },
51881     
51882     onRegionExpanded : function(region){
51883         this.fireEvent("regionexpanded", region);
51884     },
51885         
51886     /**
51887      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51888      * performs box-model adjustments.
51889      * @return {Object} The size as an object {width: (the width), height: (the height)}
51890      */
51891     getViewSize : function(){
51892         var size;
51893         if(this.el.dom != document.body){
51894             size = this.el.getSize();
51895         }else{
51896             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51897         }
51898         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51899         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51900         return size;
51901     },
51902     
51903     /**
51904      * Returns the Element this layout is bound to.
51905      * @return {Roo.Element}
51906      */
51907     getEl : function(){
51908         return this.el;
51909     },
51910     
51911     /**
51912      * Returns the specified region.
51913      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51914      * @return {Roo.LayoutRegion}
51915      */
51916     getRegion : function(target){
51917         return this.regions[target.toLowerCase()];
51918     },
51919     
51920     onWindowResize : function(){
51921         if(this.monitorWindowResize){
51922             this.layout();
51923         }
51924     }
51925 });/*
51926  * Based on:
51927  * Ext JS Library 1.1.1
51928  * Copyright(c) 2006-2007, Ext JS, LLC.
51929  *
51930  * Originally Released Under LGPL - original licence link has changed is not relivant.
51931  *
51932  * Fork - LGPL
51933  * <script type="text/javascript">
51934  */
51935 /**
51936  * @class Roo.BorderLayout
51937  * @extends Roo.LayoutManager
51938  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51939  * please see: <br><br>
51940  * <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>
51941  * <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>
51942  * Example:
51943  <pre><code>
51944  var layout = new Roo.BorderLayout(document.body, {
51945     north: {
51946         initialSize: 25,
51947         titlebar: false
51948     },
51949     west: {
51950         split:true,
51951         initialSize: 200,
51952         minSize: 175,
51953         maxSize: 400,
51954         titlebar: true,
51955         collapsible: true
51956     },
51957     east: {
51958         split:true,
51959         initialSize: 202,
51960         minSize: 175,
51961         maxSize: 400,
51962         titlebar: true,
51963         collapsible: true
51964     },
51965     south: {
51966         split:true,
51967         initialSize: 100,
51968         minSize: 100,
51969         maxSize: 200,
51970         titlebar: true,
51971         collapsible: true
51972     },
51973     center: {
51974         titlebar: true,
51975         autoScroll:true,
51976         resizeTabs: true,
51977         minTabWidth: 50,
51978         preferredTabWidth: 150
51979     }
51980 });
51981
51982 // shorthand
51983 var CP = Roo.ContentPanel;
51984
51985 layout.beginUpdate();
51986 layout.add("north", new CP("north", "North"));
51987 layout.add("south", new CP("south", {title: "South", closable: true}));
51988 layout.add("west", new CP("west", {title: "West"}));
51989 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51990 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51991 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51992 layout.getRegion("center").showPanel("center1");
51993 layout.endUpdate();
51994 </code></pre>
51995
51996 <b>The container the layout is rendered into can be either the body element or any other element.
51997 If it is not the body element, the container needs to either be an absolute positioned element,
51998 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51999 the container size if it is not the body element.</b>
52000
52001 * @constructor
52002 * Create a new BorderLayout
52003 * @param {String/HTMLElement/Element} container The container this layout is bound to
52004 * @param {Object} config Configuration options
52005  */
52006 Roo.BorderLayout = function(container, config){
52007     config = config || {};
52008     Roo.BorderLayout.superclass.constructor.call(this, container, config);
52009     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
52010     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
52011         var target = this.factory.validRegions[i];
52012         if(config[target]){
52013             this.addRegion(target, config[target]);
52014         }
52015     }
52016 };
52017
52018 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
52019     /**
52020      * Creates and adds a new region if it doesn't already exist.
52021      * @param {String} target The target region key (north, south, east, west or center).
52022      * @param {Object} config The regions config object
52023      * @return {BorderLayoutRegion} The new region
52024      */
52025     addRegion : function(target, config){
52026         if(!this.regions[target]){
52027             var r = this.factory.create(target, this, config);
52028             this.bindRegion(target, r);
52029         }
52030         return this.regions[target];
52031     },
52032
52033     // private (kinda)
52034     bindRegion : function(name, r){
52035         this.regions[name] = r;
52036         r.on("visibilitychange", this.layout, this);
52037         r.on("paneladded", this.layout, this);
52038         r.on("panelremoved", this.layout, this);
52039         r.on("invalidated", this.layout, this);
52040         r.on("resized", this.onRegionResized, this);
52041         r.on("collapsed", this.onRegionCollapsed, this);
52042         r.on("expanded", this.onRegionExpanded, this);
52043     },
52044
52045     /**
52046      * Performs a layout update.
52047      */
52048     layout : function(){
52049         if(this.updating) {
52050             return;
52051         }
52052         var size = this.getViewSize();
52053         var w = size.width;
52054         var h = size.height;
52055         var centerW = w;
52056         var centerH = h;
52057         var centerY = 0;
52058         var centerX = 0;
52059         //var x = 0, y = 0;
52060
52061         var rs = this.regions;
52062         var north = rs["north"];
52063         var south = rs["south"]; 
52064         var west = rs["west"];
52065         var east = rs["east"];
52066         var center = rs["center"];
52067         //if(this.hideOnLayout){ // not supported anymore
52068             //c.el.setStyle("display", "none");
52069         //}
52070         if(north && north.isVisible()){
52071             var b = north.getBox();
52072             var m = north.getMargins();
52073             b.width = w - (m.left+m.right);
52074             b.x = m.left;
52075             b.y = m.top;
52076             centerY = b.height + b.y + m.bottom;
52077             centerH -= centerY;
52078             north.updateBox(this.safeBox(b));
52079         }
52080         if(south && south.isVisible()){
52081             var b = south.getBox();
52082             var m = south.getMargins();
52083             b.width = w - (m.left+m.right);
52084             b.x = m.left;
52085             var totalHeight = (b.height + m.top + m.bottom);
52086             b.y = h - totalHeight + m.top;
52087             centerH -= totalHeight;
52088             south.updateBox(this.safeBox(b));
52089         }
52090         if(west && west.isVisible()){
52091             var b = west.getBox();
52092             var m = west.getMargins();
52093             b.height = centerH - (m.top+m.bottom);
52094             b.x = m.left;
52095             b.y = centerY + m.top;
52096             var totalWidth = (b.width + m.left + m.right);
52097             centerX += totalWidth;
52098             centerW -= totalWidth;
52099             west.updateBox(this.safeBox(b));
52100         }
52101         if(east && east.isVisible()){
52102             var b = east.getBox();
52103             var m = east.getMargins();
52104             b.height = centerH - (m.top+m.bottom);
52105             var totalWidth = (b.width + m.left + m.right);
52106             b.x = w - totalWidth + m.left;
52107             b.y = centerY + m.top;
52108             centerW -= totalWidth;
52109             east.updateBox(this.safeBox(b));
52110         }
52111         if(center){
52112             var m = center.getMargins();
52113             var centerBox = {
52114                 x: centerX + m.left,
52115                 y: centerY + m.top,
52116                 width: centerW - (m.left+m.right),
52117                 height: centerH - (m.top+m.bottom)
52118             };
52119             //if(this.hideOnLayout){
52120                 //center.el.setStyle("display", "block");
52121             //}
52122             center.updateBox(this.safeBox(centerBox));
52123         }
52124         this.el.repaint();
52125         this.fireEvent("layout", this);
52126     },
52127
52128     // private
52129     safeBox : function(box){
52130         box.width = Math.max(0, box.width);
52131         box.height = Math.max(0, box.height);
52132         return box;
52133     },
52134
52135     /**
52136      * Adds a ContentPanel (or subclass) to this layout.
52137      * @param {String} target The target region key (north, south, east, west or center).
52138      * @param {Roo.ContentPanel} panel The panel to add
52139      * @return {Roo.ContentPanel} The added panel
52140      */
52141     add : function(target, panel){
52142          
52143         target = target.toLowerCase();
52144         return this.regions[target].add(panel);
52145     },
52146
52147     /**
52148      * Remove a ContentPanel (or subclass) to this layout.
52149      * @param {String} target The target region key (north, south, east, west or center).
52150      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
52151      * @return {Roo.ContentPanel} The removed panel
52152      */
52153     remove : function(target, panel){
52154         target = target.toLowerCase();
52155         return this.regions[target].remove(panel);
52156     },
52157
52158     /**
52159      * Searches all regions for a panel with the specified id
52160      * @param {String} panelId
52161      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52162      */
52163     findPanel : function(panelId){
52164         var rs = this.regions;
52165         for(var target in rs){
52166             if(typeof rs[target] != "function"){
52167                 var p = rs[target].getPanel(panelId);
52168                 if(p){
52169                     return p;
52170                 }
52171             }
52172         }
52173         return null;
52174     },
52175
52176     /**
52177      * Searches all regions for a panel with the specified id and activates (shows) it.
52178      * @param {String/ContentPanel} panelId The panels id or the panel itself
52179      * @return {Roo.ContentPanel} The shown panel or null
52180      */
52181     showPanel : function(panelId) {
52182       var rs = this.regions;
52183       for(var target in rs){
52184          var r = rs[target];
52185          if(typeof r != "function"){
52186             if(r.hasPanel(panelId)){
52187                return r.showPanel(panelId);
52188             }
52189          }
52190       }
52191       return null;
52192    },
52193
52194    /**
52195      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52196      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52197      */
52198     restoreState : function(provider){
52199         if(!provider){
52200             provider = Roo.state.Manager;
52201         }
52202         var sm = new Roo.LayoutStateManager();
52203         sm.init(this, provider);
52204     },
52205
52206     /**
52207      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52208      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52209      * a valid ContentPanel config object.  Example:
52210      * <pre><code>
52211 // Create the main layout
52212 var layout = new Roo.BorderLayout('main-ct', {
52213     west: {
52214         split:true,
52215         minSize: 175,
52216         titlebar: true
52217     },
52218     center: {
52219         title:'Components'
52220     }
52221 }, 'main-ct');
52222
52223 // Create and add multiple ContentPanels at once via configs
52224 layout.batchAdd({
52225    west: {
52226        id: 'source-files',
52227        autoCreate:true,
52228        title:'Ext Source Files',
52229        autoScroll:true,
52230        fitToFrame:true
52231    },
52232    center : {
52233        el: cview,
52234        autoScroll:true,
52235        fitToFrame:true,
52236        toolbar: tb,
52237        resizeEl:'cbody'
52238    }
52239 });
52240 </code></pre>
52241      * @param {Object} regions An object containing ContentPanel configs by region name
52242      */
52243     batchAdd : function(regions){
52244         this.beginUpdate();
52245         for(var rname in regions){
52246             var lr = this.regions[rname];
52247             if(lr){
52248                 this.addTypedPanels(lr, regions[rname]);
52249             }
52250         }
52251         this.endUpdate();
52252     },
52253
52254     // private
52255     addTypedPanels : function(lr, ps){
52256         if(typeof ps == 'string'){
52257             lr.add(new Roo.ContentPanel(ps));
52258         }
52259         else if(ps instanceof Array){
52260             for(var i =0, len = ps.length; i < len; i++){
52261                 this.addTypedPanels(lr, ps[i]);
52262             }
52263         }
52264         else if(!ps.events){ // raw config?
52265             var el = ps.el;
52266             delete ps.el; // prevent conflict
52267             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52268         }
52269         else {  // panel object assumed!
52270             lr.add(ps);
52271         }
52272     },
52273     /**
52274      * Adds a xtype elements to the layout.
52275      * <pre><code>
52276
52277 layout.addxtype({
52278        xtype : 'ContentPanel',
52279        region: 'west',
52280        items: [ .... ]
52281    }
52282 );
52283
52284 layout.addxtype({
52285         xtype : 'NestedLayoutPanel',
52286         region: 'west',
52287         layout: {
52288            center: { },
52289            west: { }   
52290         },
52291         items : [ ... list of content panels or nested layout panels.. ]
52292    }
52293 );
52294 </code></pre>
52295      * @param {Object} cfg Xtype definition of item to add.
52296      */
52297     addxtype : function(cfg)
52298     {
52299         // basically accepts a pannel...
52300         // can accept a layout region..!?!?
52301         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52302         
52303         if (!cfg.xtype.match(/Panel$/)) {
52304             return false;
52305         }
52306         var ret = false;
52307         
52308         if (typeof(cfg.region) == 'undefined') {
52309             Roo.log("Failed to add Panel, region was not set");
52310             Roo.log(cfg);
52311             return false;
52312         }
52313         var region = cfg.region;
52314         delete cfg.region;
52315         
52316           
52317         var xitems = [];
52318         if (cfg.items) {
52319             xitems = cfg.items;
52320             delete cfg.items;
52321         }
52322         var nb = false;
52323         
52324         switch(cfg.xtype) 
52325         {
52326             case 'ContentPanel':  // ContentPanel (el, cfg)
52327             case 'ScrollPanel':  // ContentPanel (el, cfg)
52328             case 'ViewPanel': 
52329                 if(cfg.autoCreate) {
52330                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52331                 } else {
52332                     var el = this.el.createChild();
52333                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52334                 }
52335                 
52336                 this.add(region, ret);
52337                 break;
52338             
52339             
52340             case 'TreePanel': // our new panel!
52341                 cfg.el = this.el.createChild();
52342                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52343                 this.add(region, ret);
52344                 break;
52345             
52346             case 'NestedLayoutPanel': 
52347                 // create a new Layout (which is  a Border Layout...
52348                 var el = this.el.createChild();
52349                 var clayout = cfg.layout;
52350                 delete cfg.layout;
52351                 clayout.items   = clayout.items  || [];
52352                 // replace this exitems with the clayout ones..
52353                 xitems = clayout.items;
52354                  
52355                 
52356                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52357                     cfg.background = false;
52358                 }
52359                 var layout = new Roo.BorderLayout(el, clayout);
52360                 
52361                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52362                 //console.log('adding nested layout panel '  + cfg.toSource());
52363                 this.add(region, ret);
52364                 nb = {}; /// find first...
52365                 break;
52366                 
52367             case 'GridPanel': 
52368             
52369                 // needs grid and region
52370                 
52371                 //var el = this.getRegion(region).el.createChild();
52372                 var el = this.el.createChild();
52373                 // create the grid first...
52374                 
52375                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52376                 delete cfg.grid;
52377                 if (region == 'center' && this.active ) {
52378                     cfg.background = false;
52379                 }
52380                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52381                 
52382                 this.add(region, ret);
52383                 if (cfg.background) {
52384                     ret.on('activate', function(gp) {
52385                         if (!gp.grid.rendered) {
52386                             gp.grid.render();
52387                         }
52388                     });
52389                 } else {
52390                     grid.render();
52391                 }
52392                 break;
52393            
52394            
52395            
52396                 
52397                 
52398                 
52399             default:
52400                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52401                     
52402                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52403                     this.add(region, ret);
52404                 } else {
52405                 
52406                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52407                     return null;
52408                 }
52409                 
52410              // GridPanel (grid, cfg)
52411             
52412         }
52413         this.beginUpdate();
52414         // add children..
52415         var region = '';
52416         var abn = {};
52417         Roo.each(xitems, function(i)  {
52418             region = nb && i.region ? i.region : false;
52419             
52420             var add = ret.addxtype(i);
52421            
52422             if (region) {
52423                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52424                 if (!i.background) {
52425                     abn[region] = nb[region] ;
52426                 }
52427             }
52428             
52429         });
52430         this.endUpdate();
52431
52432         // make the last non-background panel active..
52433         //if (nb) { Roo.log(abn); }
52434         if (nb) {
52435             
52436             for(var r in abn) {
52437                 region = this.getRegion(r);
52438                 if (region) {
52439                     // tried using nb[r], but it does not work..
52440                      
52441                     region.showPanel(abn[r]);
52442                    
52443                 }
52444             }
52445         }
52446         return ret;
52447         
52448     }
52449 });
52450
52451 /**
52452  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52453  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52454  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52455  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52456  * <pre><code>
52457 // shorthand
52458 var CP = Roo.ContentPanel;
52459
52460 var layout = Roo.BorderLayout.create({
52461     north: {
52462         initialSize: 25,
52463         titlebar: false,
52464         panels: [new CP("north", "North")]
52465     },
52466     west: {
52467         split:true,
52468         initialSize: 200,
52469         minSize: 175,
52470         maxSize: 400,
52471         titlebar: true,
52472         collapsible: true,
52473         panels: [new CP("west", {title: "West"})]
52474     },
52475     east: {
52476         split:true,
52477         initialSize: 202,
52478         minSize: 175,
52479         maxSize: 400,
52480         titlebar: true,
52481         collapsible: true,
52482         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52483     },
52484     south: {
52485         split:true,
52486         initialSize: 100,
52487         minSize: 100,
52488         maxSize: 200,
52489         titlebar: true,
52490         collapsible: true,
52491         panels: [new CP("south", {title: "South", closable: true})]
52492     },
52493     center: {
52494         titlebar: true,
52495         autoScroll:true,
52496         resizeTabs: true,
52497         minTabWidth: 50,
52498         preferredTabWidth: 150,
52499         panels: [
52500             new CP("center1", {title: "Close Me", closable: true}),
52501             new CP("center2", {title: "Center Panel", closable: false})
52502         ]
52503     }
52504 }, document.body);
52505
52506 layout.getRegion("center").showPanel("center1");
52507 </code></pre>
52508  * @param config
52509  * @param targetEl
52510  */
52511 Roo.BorderLayout.create = function(config, targetEl){
52512     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52513     layout.beginUpdate();
52514     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52515     for(var j = 0, jlen = regions.length; j < jlen; j++){
52516         var lr = regions[j];
52517         if(layout.regions[lr] && config[lr].panels){
52518             var r = layout.regions[lr];
52519             var ps = config[lr].panels;
52520             layout.addTypedPanels(r, ps);
52521         }
52522     }
52523     layout.endUpdate();
52524     return layout;
52525 };
52526
52527 // private
52528 Roo.BorderLayout.RegionFactory = {
52529     // private
52530     validRegions : ["north","south","east","west","center"],
52531
52532     // private
52533     create : function(target, mgr, config){
52534         target = target.toLowerCase();
52535         if(config.lightweight || config.basic){
52536             return new Roo.BasicLayoutRegion(mgr, config, target);
52537         }
52538         switch(target){
52539             case "north":
52540                 return new Roo.NorthLayoutRegion(mgr, config);
52541             case "south":
52542                 return new Roo.SouthLayoutRegion(mgr, config);
52543             case "east":
52544                 return new Roo.EastLayoutRegion(mgr, config);
52545             case "west":
52546                 return new Roo.WestLayoutRegion(mgr, config);
52547             case "center":
52548                 return new Roo.CenterLayoutRegion(mgr, config);
52549         }
52550         throw 'Layout region "'+target+'" not supported.';
52551     }
52552 };/*
52553  * Based on:
52554  * Ext JS Library 1.1.1
52555  * Copyright(c) 2006-2007, Ext JS, LLC.
52556  *
52557  * Originally Released Under LGPL - original licence link has changed is not relivant.
52558  *
52559  * Fork - LGPL
52560  * <script type="text/javascript">
52561  */
52562  
52563 /**
52564  * @class Roo.BasicLayoutRegion
52565  * @extends Roo.util.Observable
52566  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52567  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52568  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52569  */
52570 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52571     this.mgr = mgr;
52572     this.position  = pos;
52573     this.events = {
52574         /**
52575          * @scope Roo.BasicLayoutRegion
52576          */
52577         
52578         /**
52579          * @event beforeremove
52580          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52581          * @param {Roo.LayoutRegion} this
52582          * @param {Roo.ContentPanel} panel The panel
52583          * @param {Object} e The cancel event object
52584          */
52585         "beforeremove" : true,
52586         /**
52587          * @event invalidated
52588          * Fires when the layout for this region is changed.
52589          * @param {Roo.LayoutRegion} this
52590          */
52591         "invalidated" : true,
52592         /**
52593          * @event visibilitychange
52594          * Fires when this region is shown or hidden 
52595          * @param {Roo.LayoutRegion} this
52596          * @param {Boolean} visibility true or false
52597          */
52598         "visibilitychange" : true,
52599         /**
52600          * @event paneladded
52601          * Fires when a panel is added. 
52602          * @param {Roo.LayoutRegion} this
52603          * @param {Roo.ContentPanel} panel The panel
52604          */
52605         "paneladded" : true,
52606         /**
52607          * @event panelremoved
52608          * Fires when a panel is removed. 
52609          * @param {Roo.LayoutRegion} this
52610          * @param {Roo.ContentPanel} panel The panel
52611          */
52612         "panelremoved" : true,
52613         /**
52614          * @event beforecollapse
52615          * Fires when this region before collapse.
52616          * @param {Roo.LayoutRegion} this
52617          */
52618         "beforecollapse" : true,
52619         /**
52620          * @event collapsed
52621          * Fires when this region is collapsed.
52622          * @param {Roo.LayoutRegion} this
52623          */
52624         "collapsed" : true,
52625         /**
52626          * @event expanded
52627          * Fires when this region is expanded.
52628          * @param {Roo.LayoutRegion} this
52629          */
52630         "expanded" : true,
52631         /**
52632          * @event slideshow
52633          * Fires when this region is slid into view.
52634          * @param {Roo.LayoutRegion} this
52635          */
52636         "slideshow" : true,
52637         /**
52638          * @event slidehide
52639          * Fires when this region slides out of view. 
52640          * @param {Roo.LayoutRegion} this
52641          */
52642         "slidehide" : true,
52643         /**
52644          * @event panelactivated
52645          * Fires when a panel is activated. 
52646          * @param {Roo.LayoutRegion} this
52647          * @param {Roo.ContentPanel} panel The activated panel
52648          */
52649         "panelactivated" : true,
52650         /**
52651          * @event resized
52652          * Fires when the user resizes this region. 
52653          * @param {Roo.LayoutRegion} this
52654          * @param {Number} newSize The new size (width for east/west, height for north/south)
52655          */
52656         "resized" : true
52657     };
52658     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52659     this.panels = new Roo.util.MixedCollection();
52660     this.panels.getKey = this.getPanelId.createDelegate(this);
52661     this.box = null;
52662     this.activePanel = null;
52663     // ensure listeners are added...
52664     
52665     if (config.listeners || config.events) {
52666         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52667             listeners : config.listeners || {},
52668             events : config.events || {}
52669         });
52670     }
52671     
52672     if(skipConfig !== true){
52673         this.applyConfig(config);
52674     }
52675 };
52676
52677 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52678     getPanelId : function(p){
52679         return p.getId();
52680     },
52681     
52682     applyConfig : function(config){
52683         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52684         this.config = config;
52685         
52686     },
52687     
52688     /**
52689      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52690      * the width, for horizontal (north, south) the height.
52691      * @param {Number} newSize The new width or height
52692      */
52693     resizeTo : function(newSize){
52694         var el = this.el ? this.el :
52695                  (this.activePanel ? this.activePanel.getEl() : null);
52696         if(el){
52697             switch(this.position){
52698                 case "east":
52699                 case "west":
52700                     el.setWidth(newSize);
52701                     this.fireEvent("resized", this, newSize);
52702                 break;
52703                 case "north":
52704                 case "south":
52705                     el.setHeight(newSize);
52706                     this.fireEvent("resized", this, newSize);
52707                 break;                
52708             }
52709         }
52710     },
52711     
52712     getBox : function(){
52713         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52714     },
52715     
52716     getMargins : function(){
52717         return this.margins;
52718     },
52719     
52720     updateBox : function(box){
52721         this.box = box;
52722         var el = this.activePanel.getEl();
52723         el.dom.style.left = box.x + "px";
52724         el.dom.style.top = box.y + "px";
52725         this.activePanel.setSize(box.width, box.height);
52726     },
52727     
52728     /**
52729      * Returns the container element for this region.
52730      * @return {Roo.Element}
52731      */
52732     getEl : function(){
52733         return this.activePanel;
52734     },
52735     
52736     /**
52737      * Returns true if this region is currently visible.
52738      * @return {Boolean}
52739      */
52740     isVisible : function(){
52741         return this.activePanel ? true : false;
52742     },
52743     
52744     setActivePanel : function(panel){
52745         panel = this.getPanel(panel);
52746         if(this.activePanel && this.activePanel != panel){
52747             this.activePanel.setActiveState(false);
52748             this.activePanel.getEl().setLeftTop(-10000,-10000);
52749         }
52750         this.activePanel = panel;
52751         panel.setActiveState(true);
52752         if(this.box){
52753             panel.setSize(this.box.width, this.box.height);
52754         }
52755         this.fireEvent("panelactivated", this, panel);
52756         this.fireEvent("invalidated");
52757     },
52758     
52759     /**
52760      * Show the specified panel.
52761      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52762      * @return {Roo.ContentPanel} The shown panel or null
52763      */
52764     showPanel : function(panel){
52765         if(panel = this.getPanel(panel)){
52766             this.setActivePanel(panel);
52767         }
52768         return panel;
52769     },
52770     
52771     /**
52772      * Get the active panel for this region.
52773      * @return {Roo.ContentPanel} The active panel or null
52774      */
52775     getActivePanel : function(){
52776         return this.activePanel;
52777     },
52778     
52779     /**
52780      * Add the passed ContentPanel(s)
52781      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52782      * @return {Roo.ContentPanel} The panel added (if only one was added)
52783      */
52784     add : function(panel){
52785         if(arguments.length > 1){
52786             for(var i = 0, len = arguments.length; i < len; i++) {
52787                 this.add(arguments[i]);
52788             }
52789             return null;
52790         }
52791         if(this.hasPanel(panel)){
52792             this.showPanel(panel);
52793             return panel;
52794         }
52795         var el = panel.getEl();
52796         if(el.dom.parentNode != this.mgr.el.dom){
52797             this.mgr.el.dom.appendChild(el.dom);
52798         }
52799         if(panel.setRegion){
52800             panel.setRegion(this);
52801         }
52802         this.panels.add(panel);
52803         el.setStyle("position", "absolute");
52804         if(!panel.background){
52805             this.setActivePanel(panel);
52806             if(this.config.initialSize && this.panels.getCount()==1){
52807                 this.resizeTo(this.config.initialSize);
52808             }
52809         }
52810         this.fireEvent("paneladded", this, panel);
52811         return panel;
52812     },
52813     
52814     /**
52815      * Returns true if the panel is in this region.
52816      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52817      * @return {Boolean}
52818      */
52819     hasPanel : function(panel){
52820         if(typeof panel == "object"){ // must be panel obj
52821             panel = panel.getId();
52822         }
52823         return this.getPanel(panel) ? true : false;
52824     },
52825     
52826     /**
52827      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52828      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52829      * @param {Boolean} preservePanel Overrides the config preservePanel option
52830      * @return {Roo.ContentPanel} The panel that was removed
52831      */
52832     remove : function(panel, preservePanel){
52833         panel = this.getPanel(panel);
52834         if(!panel){
52835             return null;
52836         }
52837         var e = {};
52838         this.fireEvent("beforeremove", this, panel, e);
52839         if(e.cancel === true){
52840             return null;
52841         }
52842         var panelId = panel.getId();
52843         this.panels.removeKey(panelId);
52844         return panel;
52845     },
52846     
52847     /**
52848      * Returns the panel specified or null if it's not in this region.
52849      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52850      * @return {Roo.ContentPanel}
52851      */
52852     getPanel : function(id){
52853         if(typeof id == "object"){ // must be panel obj
52854             return id;
52855         }
52856         return this.panels.get(id);
52857     },
52858     
52859     /**
52860      * Returns this regions position (north/south/east/west/center).
52861      * @return {String} 
52862      */
52863     getPosition: function(){
52864         return this.position;    
52865     }
52866 });/*
52867  * Based on:
52868  * Ext JS Library 1.1.1
52869  * Copyright(c) 2006-2007, Ext JS, LLC.
52870  *
52871  * Originally Released Under LGPL - original licence link has changed is not relivant.
52872  *
52873  * Fork - LGPL
52874  * <script type="text/javascript">
52875  */
52876  
52877 /**
52878  * @class Roo.LayoutRegion
52879  * @extends Roo.BasicLayoutRegion
52880  * This class represents a region in a layout manager.
52881  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52882  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52883  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52884  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52885  * @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})
52886  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52887  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52888  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52889  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52890  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52891  * @cfg {String}    title           The title for the region (overrides panel titles)
52892  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52893  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52894  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52895  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52896  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52897  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52898  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52899  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52900  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52901  * @cfg {Boolean}   showPin         True to show a pin button
52902  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52903  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52904  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52905  * @cfg {Number}    width           For East/West panels
52906  * @cfg {Number}    height          For North/South panels
52907  * @cfg {Boolean}   split           To show the splitter
52908  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52909  */
52910 Roo.LayoutRegion = function(mgr, config, pos){
52911     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52912     var dh = Roo.DomHelper;
52913     /** This region's container element 
52914     * @type Roo.Element */
52915     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52916     /** This region's title element 
52917     * @type Roo.Element */
52918
52919     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52920         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52921         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52922     ]}, true);
52923     this.titleEl.enableDisplayMode();
52924     /** This region's title text element 
52925     * @type HTMLElement */
52926     this.titleTextEl = this.titleEl.dom.firstChild;
52927     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52928     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52929     this.closeBtn.enableDisplayMode();
52930     this.closeBtn.on("click", this.closeClicked, this);
52931     this.closeBtn.hide();
52932
52933     this.createBody(config);
52934     this.visible = true;
52935     this.collapsed = false;
52936
52937     if(config.hideWhenEmpty){
52938         this.hide();
52939         this.on("paneladded", this.validateVisibility, this);
52940         this.on("panelremoved", this.validateVisibility, this);
52941     }
52942     this.applyConfig(config);
52943 };
52944
52945 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52946
52947     createBody : function(){
52948         /** This region's body element 
52949         * @type Roo.Element */
52950         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52951     },
52952
52953     applyConfig : function(c){
52954         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52955             var dh = Roo.DomHelper;
52956             if(c.titlebar !== false){
52957                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52958                 this.collapseBtn.on("click", this.collapse, this);
52959                 this.collapseBtn.enableDisplayMode();
52960
52961                 if(c.showPin === true || this.showPin){
52962                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52963                     this.stickBtn.enableDisplayMode();
52964                     this.stickBtn.on("click", this.expand, this);
52965                     this.stickBtn.hide();
52966                 }
52967             }
52968             /** This region's collapsed element
52969             * @type Roo.Element */
52970             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52971                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52972             ]}, true);
52973             if(c.floatable !== false){
52974                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52975                this.collapsedEl.on("click", this.collapseClick, this);
52976             }
52977
52978             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52979                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52980                    id: "message", unselectable: "on", style:{"float":"left"}});
52981                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52982              }
52983             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52984             this.expandBtn.on("click", this.expand, this);
52985         }
52986         if(this.collapseBtn){
52987             this.collapseBtn.setVisible(c.collapsible == true);
52988         }
52989         this.cmargins = c.cmargins || this.cmargins ||
52990                          (this.position == "west" || this.position == "east" ?
52991                              {top: 0, left: 2, right:2, bottom: 0} :
52992                              {top: 2, left: 0, right:0, bottom: 2});
52993         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52994         this.bottomTabs = c.tabPosition != "top";
52995         this.autoScroll = c.autoScroll || false;
52996         if(this.autoScroll){
52997             this.bodyEl.setStyle("overflow", "auto");
52998         }else{
52999             this.bodyEl.setStyle("overflow", "hidden");
53000         }
53001         //if(c.titlebar !== false){
53002             if((!c.titlebar && !c.title) || c.titlebar === false){
53003                 this.titleEl.hide();
53004             }else{
53005                 this.titleEl.show();
53006                 if(c.title){
53007                     this.titleTextEl.innerHTML = c.title;
53008                 }
53009             }
53010         //}
53011         this.duration = c.duration || .30;
53012         this.slideDuration = c.slideDuration || .45;
53013         this.config = c;
53014         if(c.collapsed){
53015             this.collapse(true);
53016         }
53017         if(c.hidden){
53018             this.hide();
53019         }
53020     },
53021     /**
53022      * Returns true if this region is currently visible.
53023      * @return {Boolean}
53024      */
53025     isVisible : function(){
53026         return this.visible;
53027     },
53028
53029     /**
53030      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
53031      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
53032      */
53033     setCollapsedTitle : function(title){
53034         title = title || "&#160;";
53035         if(this.collapsedTitleTextEl){
53036             this.collapsedTitleTextEl.innerHTML = title;
53037         }
53038     },
53039
53040     getBox : function(){
53041         var b;
53042         if(!this.collapsed){
53043             b = this.el.getBox(false, true);
53044         }else{
53045             b = this.collapsedEl.getBox(false, true);
53046         }
53047         return b;
53048     },
53049
53050     getMargins : function(){
53051         return this.collapsed ? this.cmargins : this.margins;
53052     },
53053
53054     highlight : function(){
53055         this.el.addClass("x-layout-panel-dragover");
53056     },
53057
53058     unhighlight : function(){
53059         this.el.removeClass("x-layout-panel-dragover");
53060     },
53061
53062     updateBox : function(box){
53063         this.box = box;
53064         if(!this.collapsed){
53065             this.el.dom.style.left = box.x + "px";
53066             this.el.dom.style.top = box.y + "px";
53067             this.updateBody(box.width, box.height);
53068         }else{
53069             this.collapsedEl.dom.style.left = box.x + "px";
53070             this.collapsedEl.dom.style.top = box.y + "px";
53071             this.collapsedEl.setSize(box.width, box.height);
53072         }
53073         if(this.tabs){
53074             this.tabs.autoSizeTabs();
53075         }
53076     },
53077
53078     updateBody : function(w, h){
53079         if(w !== null){
53080             this.el.setWidth(w);
53081             w -= this.el.getBorderWidth("rl");
53082             if(this.config.adjustments){
53083                 w += this.config.adjustments[0];
53084             }
53085         }
53086         if(h !== null){
53087             this.el.setHeight(h);
53088             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
53089             h -= this.el.getBorderWidth("tb");
53090             if(this.config.adjustments){
53091                 h += this.config.adjustments[1];
53092             }
53093             this.bodyEl.setHeight(h);
53094             if(this.tabs){
53095                 h = this.tabs.syncHeight(h);
53096             }
53097         }
53098         if(this.panelSize){
53099             w = w !== null ? w : this.panelSize.width;
53100             h = h !== null ? h : this.panelSize.height;
53101         }
53102         if(this.activePanel){
53103             var el = this.activePanel.getEl();
53104             w = w !== null ? w : el.getWidth();
53105             h = h !== null ? h : el.getHeight();
53106             this.panelSize = {width: w, height: h};
53107             this.activePanel.setSize(w, h);
53108         }
53109         if(Roo.isIE && this.tabs){
53110             this.tabs.el.repaint();
53111         }
53112     },
53113
53114     /**
53115      * Returns the container element for this region.
53116      * @return {Roo.Element}
53117      */
53118     getEl : function(){
53119         return this.el;
53120     },
53121
53122     /**
53123      * Hides this region.
53124      */
53125     hide : function(){
53126         if(!this.collapsed){
53127             this.el.dom.style.left = "-2000px";
53128             this.el.hide();
53129         }else{
53130             this.collapsedEl.dom.style.left = "-2000px";
53131             this.collapsedEl.hide();
53132         }
53133         this.visible = false;
53134         this.fireEvent("visibilitychange", this, false);
53135     },
53136
53137     /**
53138      * Shows this region if it was previously hidden.
53139      */
53140     show : function(){
53141         if(!this.collapsed){
53142             this.el.show();
53143         }else{
53144             this.collapsedEl.show();
53145         }
53146         this.visible = true;
53147         this.fireEvent("visibilitychange", this, true);
53148     },
53149
53150     closeClicked : function(){
53151         if(this.activePanel){
53152             this.remove(this.activePanel);
53153         }
53154     },
53155
53156     collapseClick : function(e){
53157         if(this.isSlid){
53158            e.stopPropagation();
53159            this.slideIn();
53160         }else{
53161            e.stopPropagation();
53162            this.slideOut();
53163         }
53164     },
53165
53166     /**
53167      * Collapses this region.
53168      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53169      */
53170     collapse : function(skipAnim, skipCheck){
53171         if(this.collapsed) {
53172             return;
53173         }
53174         
53175         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53176             
53177             this.collapsed = true;
53178             if(this.split){
53179                 this.split.el.hide();
53180             }
53181             if(this.config.animate && skipAnim !== true){
53182                 this.fireEvent("invalidated", this);
53183                 this.animateCollapse();
53184             }else{
53185                 this.el.setLocation(-20000,-20000);
53186                 this.el.hide();
53187                 this.collapsedEl.show();
53188                 this.fireEvent("collapsed", this);
53189                 this.fireEvent("invalidated", this);
53190             }
53191         }
53192         
53193     },
53194
53195     animateCollapse : function(){
53196         // overridden
53197     },
53198
53199     /**
53200      * Expands this region if it was previously collapsed.
53201      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53202      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53203      */
53204     expand : function(e, skipAnim){
53205         if(e) {
53206             e.stopPropagation();
53207         }
53208         if(!this.collapsed || this.el.hasActiveFx()) {
53209             return;
53210         }
53211         if(this.isSlid){
53212             this.afterSlideIn();
53213             skipAnim = true;
53214         }
53215         this.collapsed = false;
53216         if(this.config.animate && skipAnim !== true){
53217             this.animateExpand();
53218         }else{
53219             this.el.show();
53220             if(this.split){
53221                 this.split.el.show();
53222             }
53223             this.collapsedEl.setLocation(-2000,-2000);
53224             this.collapsedEl.hide();
53225             this.fireEvent("invalidated", this);
53226             this.fireEvent("expanded", this);
53227         }
53228     },
53229
53230     animateExpand : function(){
53231         // overridden
53232     },
53233
53234     initTabs : function()
53235     {
53236         this.bodyEl.setStyle("overflow", "hidden");
53237         var ts = new Roo.TabPanel(
53238                 this.bodyEl.dom,
53239                 {
53240                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53241                     disableTooltips: this.config.disableTabTips,
53242                     toolbar : this.config.toolbar
53243                 }
53244         );
53245         if(this.config.hideTabs){
53246             ts.stripWrap.setDisplayed(false);
53247         }
53248         this.tabs = ts;
53249         ts.resizeTabs = this.config.resizeTabs === true;
53250         ts.minTabWidth = this.config.minTabWidth || 40;
53251         ts.maxTabWidth = this.config.maxTabWidth || 250;
53252         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53253         ts.monitorResize = false;
53254         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53255         ts.bodyEl.addClass('x-layout-tabs-body');
53256         this.panels.each(this.initPanelAsTab, this);
53257     },
53258
53259     initPanelAsTab : function(panel){
53260         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53261                     this.config.closeOnTab && panel.isClosable());
53262         if(panel.tabTip !== undefined){
53263             ti.setTooltip(panel.tabTip);
53264         }
53265         ti.on("activate", function(){
53266               this.setActivePanel(panel);
53267         }, this);
53268         if(this.config.closeOnTab){
53269             ti.on("beforeclose", function(t, e){
53270                 e.cancel = true;
53271                 this.remove(panel);
53272             }, this);
53273         }
53274         return ti;
53275     },
53276
53277     updatePanelTitle : function(panel, title){
53278         if(this.activePanel == panel){
53279             this.updateTitle(title);
53280         }
53281         if(this.tabs){
53282             var ti = this.tabs.getTab(panel.getEl().id);
53283             ti.setText(title);
53284             if(panel.tabTip !== undefined){
53285                 ti.setTooltip(panel.tabTip);
53286             }
53287         }
53288     },
53289
53290     updateTitle : function(title){
53291         if(this.titleTextEl && !this.config.title){
53292             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53293         }
53294     },
53295
53296     setActivePanel : function(panel){
53297         panel = this.getPanel(panel);
53298         if(this.activePanel && this.activePanel != panel){
53299             this.activePanel.setActiveState(false);
53300         }
53301         this.activePanel = panel;
53302         panel.setActiveState(true);
53303         if(this.panelSize){
53304             panel.setSize(this.panelSize.width, this.panelSize.height);
53305         }
53306         if(this.closeBtn){
53307             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53308         }
53309         this.updateTitle(panel.getTitle());
53310         if(this.tabs){
53311             this.fireEvent("invalidated", this);
53312         }
53313         this.fireEvent("panelactivated", this, panel);
53314     },
53315
53316     /**
53317      * Shows the specified panel.
53318      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53319      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53320      */
53321     showPanel : function(panel)
53322     {
53323         panel = this.getPanel(panel);
53324         if(panel){
53325             if(this.tabs){
53326                 var tab = this.tabs.getTab(panel.getEl().id);
53327                 if(tab.isHidden()){
53328                     this.tabs.unhideTab(tab.id);
53329                 }
53330                 tab.activate();
53331             }else{
53332                 this.setActivePanel(panel);
53333             }
53334         }
53335         return panel;
53336     },
53337
53338     /**
53339      * Get the active panel for this region.
53340      * @return {Roo.ContentPanel} The active panel or null
53341      */
53342     getActivePanel : function(){
53343         return this.activePanel;
53344     },
53345
53346     validateVisibility : function(){
53347         if(this.panels.getCount() < 1){
53348             this.updateTitle("&#160;");
53349             this.closeBtn.hide();
53350             this.hide();
53351         }else{
53352             if(!this.isVisible()){
53353                 this.show();
53354             }
53355         }
53356     },
53357
53358     /**
53359      * Adds the passed ContentPanel(s) to this region.
53360      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53361      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53362      */
53363     add : function(panel){
53364         if(arguments.length > 1){
53365             for(var i = 0, len = arguments.length; i < len; i++) {
53366                 this.add(arguments[i]);
53367             }
53368             return null;
53369         }
53370         if(this.hasPanel(panel)){
53371             this.showPanel(panel);
53372             return panel;
53373         }
53374         panel.setRegion(this);
53375         this.panels.add(panel);
53376         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53377             this.bodyEl.dom.appendChild(panel.getEl().dom);
53378             if(panel.background !== true){
53379                 this.setActivePanel(panel);
53380             }
53381             this.fireEvent("paneladded", this, panel);
53382             return panel;
53383         }
53384         if(!this.tabs){
53385             this.initTabs();
53386         }else{
53387             this.initPanelAsTab(panel);
53388         }
53389         if(panel.background !== true){
53390             this.tabs.activate(panel.getEl().id);
53391         }
53392         this.fireEvent("paneladded", this, panel);
53393         return panel;
53394     },
53395
53396     /**
53397      * Hides the tab for the specified panel.
53398      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53399      */
53400     hidePanel : function(panel){
53401         if(this.tabs && (panel = this.getPanel(panel))){
53402             this.tabs.hideTab(panel.getEl().id);
53403         }
53404     },
53405
53406     /**
53407      * Unhides the tab for a previously hidden panel.
53408      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53409      */
53410     unhidePanel : function(panel){
53411         if(this.tabs && (panel = this.getPanel(panel))){
53412             this.tabs.unhideTab(panel.getEl().id);
53413         }
53414     },
53415
53416     clearPanels : function(){
53417         while(this.panels.getCount() > 0){
53418              this.remove(this.panels.first());
53419         }
53420     },
53421
53422     /**
53423      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53424      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53425      * @param {Boolean} preservePanel Overrides the config preservePanel option
53426      * @return {Roo.ContentPanel} The panel that was removed
53427      */
53428     remove : function(panel, preservePanel){
53429         panel = this.getPanel(panel);
53430         if(!panel){
53431             return null;
53432         }
53433         var e = {};
53434         this.fireEvent("beforeremove", this, panel, e);
53435         if(e.cancel === true){
53436             return null;
53437         }
53438         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53439         var panelId = panel.getId();
53440         this.panels.removeKey(panelId);
53441         if(preservePanel){
53442             document.body.appendChild(panel.getEl().dom);
53443         }
53444         if(this.tabs){
53445             this.tabs.removeTab(panel.getEl().id);
53446         }else if (!preservePanel){
53447             this.bodyEl.dom.removeChild(panel.getEl().dom);
53448         }
53449         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53450             var p = this.panels.first();
53451             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53452             tempEl.appendChild(p.getEl().dom);
53453             this.bodyEl.update("");
53454             this.bodyEl.dom.appendChild(p.getEl().dom);
53455             tempEl = null;
53456             this.updateTitle(p.getTitle());
53457             this.tabs = null;
53458             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53459             this.setActivePanel(p);
53460         }
53461         panel.setRegion(null);
53462         if(this.activePanel == panel){
53463             this.activePanel = null;
53464         }
53465         if(this.config.autoDestroy !== false && preservePanel !== true){
53466             try{panel.destroy();}catch(e){}
53467         }
53468         this.fireEvent("panelremoved", this, panel);
53469         return panel;
53470     },
53471
53472     /**
53473      * Returns the TabPanel component used by this region
53474      * @return {Roo.TabPanel}
53475      */
53476     getTabs : function(){
53477         return this.tabs;
53478     },
53479
53480     createTool : function(parentEl, className){
53481         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53482             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53483         btn.addClassOnOver("x-layout-tools-button-over");
53484         return btn;
53485     }
53486 });/*
53487  * Based on:
53488  * Ext JS Library 1.1.1
53489  * Copyright(c) 2006-2007, Ext JS, LLC.
53490  *
53491  * Originally Released Under LGPL - original licence link has changed is not relivant.
53492  *
53493  * Fork - LGPL
53494  * <script type="text/javascript">
53495  */
53496  
53497
53498
53499 /**
53500  * @class Roo.SplitLayoutRegion
53501  * @extends Roo.LayoutRegion
53502  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53503  */
53504 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53505     this.cursor = cursor;
53506     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53507 };
53508
53509 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53510     splitTip : "Drag to resize.",
53511     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53512     useSplitTips : false,
53513
53514     applyConfig : function(config){
53515         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53516         if(config.split){
53517             if(!this.split){
53518                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53519                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53520                 /** The SplitBar for this region 
53521                 * @type Roo.SplitBar */
53522                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53523                 this.split.on("moved", this.onSplitMove, this);
53524                 this.split.useShim = config.useShim === true;
53525                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53526                 if(this.useSplitTips){
53527                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53528                 }
53529                 if(config.collapsible){
53530                     this.split.el.on("dblclick", this.collapse,  this);
53531                 }
53532             }
53533             if(typeof config.minSize != "undefined"){
53534                 this.split.minSize = config.minSize;
53535             }
53536             if(typeof config.maxSize != "undefined"){
53537                 this.split.maxSize = config.maxSize;
53538             }
53539             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53540                 this.hideSplitter();
53541             }
53542         }
53543     },
53544
53545     getHMaxSize : function(){
53546          var cmax = this.config.maxSize || 10000;
53547          var center = this.mgr.getRegion("center");
53548          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53549     },
53550
53551     getVMaxSize : function(){
53552          var cmax = this.config.maxSize || 10000;
53553          var center = this.mgr.getRegion("center");
53554          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53555     },
53556
53557     onSplitMove : function(split, newSize){
53558         this.fireEvent("resized", this, newSize);
53559     },
53560     
53561     /** 
53562      * Returns the {@link Roo.SplitBar} for this region.
53563      * @return {Roo.SplitBar}
53564      */
53565     getSplitBar : function(){
53566         return this.split;
53567     },
53568     
53569     hide : function(){
53570         this.hideSplitter();
53571         Roo.SplitLayoutRegion.superclass.hide.call(this);
53572     },
53573
53574     hideSplitter : function(){
53575         if(this.split){
53576             this.split.el.setLocation(-2000,-2000);
53577             this.split.el.hide();
53578         }
53579     },
53580
53581     show : function(){
53582         if(this.split){
53583             this.split.el.show();
53584         }
53585         Roo.SplitLayoutRegion.superclass.show.call(this);
53586     },
53587     
53588     beforeSlide: function(){
53589         if(Roo.isGecko){// firefox overflow auto bug workaround
53590             this.bodyEl.clip();
53591             if(this.tabs) {
53592                 this.tabs.bodyEl.clip();
53593             }
53594             if(this.activePanel){
53595                 this.activePanel.getEl().clip();
53596                 
53597                 if(this.activePanel.beforeSlide){
53598                     this.activePanel.beforeSlide();
53599                 }
53600             }
53601         }
53602     },
53603     
53604     afterSlide : function(){
53605         if(Roo.isGecko){// firefox overflow auto bug workaround
53606             this.bodyEl.unclip();
53607             if(this.tabs) {
53608                 this.tabs.bodyEl.unclip();
53609             }
53610             if(this.activePanel){
53611                 this.activePanel.getEl().unclip();
53612                 if(this.activePanel.afterSlide){
53613                     this.activePanel.afterSlide();
53614                 }
53615             }
53616         }
53617     },
53618
53619     initAutoHide : function(){
53620         if(this.autoHide !== false){
53621             if(!this.autoHideHd){
53622                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53623                 this.autoHideHd = {
53624                     "mouseout": function(e){
53625                         if(!e.within(this.el, true)){
53626                             st.delay(500);
53627                         }
53628                     },
53629                     "mouseover" : function(e){
53630                         st.cancel();
53631                     },
53632                     scope : this
53633                 };
53634             }
53635             this.el.on(this.autoHideHd);
53636         }
53637     },
53638
53639     clearAutoHide : function(){
53640         if(this.autoHide !== false){
53641             this.el.un("mouseout", this.autoHideHd.mouseout);
53642             this.el.un("mouseover", this.autoHideHd.mouseover);
53643         }
53644     },
53645
53646     clearMonitor : function(){
53647         Roo.get(document).un("click", this.slideInIf, this);
53648     },
53649
53650     // these names are backwards but not changed for compat
53651     slideOut : function(){
53652         if(this.isSlid || this.el.hasActiveFx()){
53653             return;
53654         }
53655         this.isSlid = true;
53656         if(this.collapseBtn){
53657             this.collapseBtn.hide();
53658         }
53659         this.closeBtnState = this.closeBtn.getStyle('display');
53660         this.closeBtn.hide();
53661         if(this.stickBtn){
53662             this.stickBtn.show();
53663         }
53664         this.el.show();
53665         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53666         this.beforeSlide();
53667         this.el.setStyle("z-index", 10001);
53668         this.el.slideIn(this.getSlideAnchor(), {
53669             callback: function(){
53670                 this.afterSlide();
53671                 this.initAutoHide();
53672                 Roo.get(document).on("click", this.slideInIf, this);
53673                 this.fireEvent("slideshow", this);
53674             },
53675             scope: this,
53676             block: true
53677         });
53678     },
53679
53680     afterSlideIn : function(){
53681         this.clearAutoHide();
53682         this.isSlid = false;
53683         this.clearMonitor();
53684         this.el.setStyle("z-index", "");
53685         if(this.collapseBtn){
53686             this.collapseBtn.show();
53687         }
53688         this.closeBtn.setStyle('display', this.closeBtnState);
53689         if(this.stickBtn){
53690             this.stickBtn.hide();
53691         }
53692         this.fireEvent("slidehide", this);
53693     },
53694
53695     slideIn : function(cb){
53696         if(!this.isSlid || this.el.hasActiveFx()){
53697             Roo.callback(cb);
53698             return;
53699         }
53700         this.isSlid = false;
53701         this.beforeSlide();
53702         this.el.slideOut(this.getSlideAnchor(), {
53703             callback: function(){
53704                 this.el.setLeftTop(-10000, -10000);
53705                 this.afterSlide();
53706                 this.afterSlideIn();
53707                 Roo.callback(cb);
53708             },
53709             scope: this,
53710             block: true
53711         });
53712     },
53713     
53714     slideInIf : function(e){
53715         if(!e.within(this.el)){
53716             this.slideIn();
53717         }
53718     },
53719
53720     animateCollapse : function(){
53721         this.beforeSlide();
53722         this.el.setStyle("z-index", 20000);
53723         var anchor = this.getSlideAnchor();
53724         this.el.slideOut(anchor, {
53725             callback : function(){
53726                 this.el.setStyle("z-index", "");
53727                 this.collapsedEl.slideIn(anchor, {duration:.3});
53728                 this.afterSlide();
53729                 this.el.setLocation(-10000,-10000);
53730                 this.el.hide();
53731                 this.fireEvent("collapsed", this);
53732             },
53733             scope: this,
53734             block: true
53735         });
53736     },
53737
53738     animateExpand : function(){
53739         this.beforeSlide();
53740         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53741         this.el.setStyle("z-index", 20000);
53742         this.collapsedEl.hide({
53743             duration:.1
53744         });
53745         this.el.slideIn(this.getSlideAnchor(), {
53746             callback : function(){
53747                 this.el.setStyle("z-index", "");
53748                 this.afterSlide();
53749                 if(this.split){
53750                     this.split.el.show();
53751                 }
53752                 this.fireEvent("invalidated", this);
53753                 this.fireEvent("expanded", this);
53754             },
53755             scope: this,
53756             block: true
53757         });
53758     },
53759
53760     anchors : {
53761         "west" : "left",
53762         "east" : "right",
53763         "north" : "top",
53764         "south" : "bottom"
53765     },
53766
53767     sanchors : {
53768         "west" : "l",
53769         "east" : "r",
53770         "north" : "t",
53771         "south" : "b"
53772     },
53773
53774     canchors : {
53775         "west" : "tl-tr",
53776         "east" : "tr-tl",
53777         "north" : "tl-bl",
53778         "south" : "bl-tl"
53779     },
53780
53781     getAnchor : function(){
53782         return this.anchors[this.position];
53783     },
53784
53785     getCollapseAnchor : function(){
53786         return this.canchors[this.position];
53787     },
53788
53789     getSlideAnchor : function(){
53790         return this.sanchors[this.position];
53791     },
53792
53793     getAlignAdj : function(){
53794         var cm = this.cmargins;
53795         switch(this.position){
53796             case "west":
53797                 return [0, 0];
53798             break;
53799             case "east":
53800                 return [0, 0];
53801             break;
53802             case "north":
53803                 return [0, 0];
53804             break;
53805             case "south":
53806                 return [0, 0];
53807             break;
53808         }
53809     },
53810
53811     getExpandAdj : function(){
53812         var c = this.collapsedEl, cm = this.cmargins;
53813         switch(this.position){
53814             case "west":
53815                 return [-(cm.right+c.getWidth()+cm.left), 0];
53816             break;
53817             case "east":
53818                 return [cm.right+c.getWidth()+cm.left, 0];
53819             break;
53820             case "north":
53821                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53822             break;
53823             case "south":
53824                 return [0, cm.top+cm.bottom+c.getHeight()];
53825             break;
53826         }
53827     }
53828 });/*
53829  * Based on:
53830  * Ext JS Library 1.1.1
53831  * Copyright(c) 2006-2007, Ext JS, LLC.
53832  *
53833  * Originally Released Under LGPL - original licence link has changed is not relivant.
53834  *
53835  * Fork - LGPL
53836  * <script type="text/javascript">
53837  */
53838 /*
53839  * These classes are private internal classes
53840  */
53841 Roo.CenterLayoutRegion = function(mgr, config){
53842     Roo.LayoutRegion.call(this, mgr, config, "center");
53843     this.visible = true;
53844     this.minWidth = config.minWidth || 20;
53845     this.minHeight = config.minHeight || 20;
53846 };
53847
53848 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53849     hide : function(){
53850         // center panel can't be hidden
53851     },
53852     
53853     show : function(){
53854         // center panel can't be hidden
53855     },
53856     
53857     getMinWidth: function(){
53858         return this.minWidth;
53859     },
53860     
53861     getMinHeight: function(){
53862         return this.minHeight;
53863     }
53864 });
53865
53866
53867 Roo.NorthLayoutRegion = function(mgr, config){
53868     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53869     if(this.split){
53870         this.split.placement = Roo.SplitBar.TOP;
53871         this.split.orientation = Roo.SplitBar.VERTICAL;
53872         this.split.el.addClass("x-layout-split-v");
53873     }
53874     var size = config.initialSize || config.height;
53875     if(typeof size != "undefined"){
53876         this.el.setHeight(size);
53877     }
53878 };
53879 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53880     orientation: Roo.SplitBar.VERTICAL,
53881     getBox : function(){
53882         if(this.collapsed){
53883             return this.collapsedEl.getBox();
53884         }
53885         var box = this.el.getBox();
53886         if(this.split){
53887             box.height += this.split.el.getHeight();
53888         }
53889         return box;
53890     },
53891     
53892     updateBox : function(box){
53893         if(this.split && !this.collapsed){
53894             box.height -= this.split.el.getHeight();
53895             this.split.el.setLeft(box.x);
53896             this.split.el.setTop(box.y+box.height);
53897             this.split.el.setWidth(box.width);
53898         }
53899         if(this.collapsed){
53900             this.updateBody(box.width, null);
53901         }
53902         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53903     }
53904 });
53905
53906 Roo.SouthLayoutRegion = function(mgr, config){
53907     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53908     if(this.split){
53909         this.split.placement = Roo.SplitBar.BOTTOM;
53910         this.split.orientation = Roo.SplitBar.VERTICAL;
53911         this.split.el.addClass("x-layout-split-v");
53912     }
53913     var size = config.initialSize || config.height;
53914     if(typeof size != "undefined"){
53915         this.el.setHeight(size);
53916     }
53917 };
53918 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53919     orientation: Roo.SplitBar.VERTICAL,
53920     getBox : function(){
53921         if(this.collapsed){
53922             return this.collapsedEl.getBox();
53923         }
53924         var box = this.el.getBox();
53925         if(this.split){
53926             var sh = this.split.el.getHeight();
53927             box.height += sh;
53928             box.y -= sh;
53929         }
53930         return box;
53931     },
53932     
53933     updateBox : function(box){
53934         if(this.split && !this.collapsed){
53935             var sh = this.split.el.getHeight();
53936             box.height -= sh;
53937             box.y += sh;
53938             this.split.el.setLeft(box.x);
53939             this.split.el.setTop(box.y-sh);
53940             this.split.el.setWidth(box.width);
53941         }
53942         if(this.collapsed){
53943             this.updateBody(box.width, null);
53944         }
53945         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53946     }
53947 });
53948
53949 Roo.EastLayoutRegion = function(mgr, config){
53950     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53951     if(this.split){
53952         this.split.placement = Roo.SplitBar.RIGHT;
53953         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53954         this.split.el.addClass("x-layout-split-h");
53955     }
53956     var size = config.initialSize || config.width;
53957     if(typeof size != "undefined"){
53958         this.el.setWidth(size);
53959     }
53960 };
53961 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53962     orientation: Roo.SplitBar.HORIZONTAL,
53963     getBox : function(){
53964         if(this.collapsed){
53965             return this.collapsedEl.getBox();
53966         }
53967         var box = this.el.getBox();
53968         if(this.split){
53969             var sw = this.split.el.getWidth();
53970             box.width += sw;
53971             box.x -= sw;
53972         }
53973         return box;
53974     },
53975
53976     updateBox : function(box){
53977         if(this.split && !this.collapsed){
53978             var sw = this.split.el.getWidth();
53979             box.width -= sw;
53980             this.split.el.setLeft(box.x);
53981             this.split.el.setTop(box.y);
53982             this.split.el.setHeight(box.height);
53983             box.x += sw;
53984         }
53985         if(this.collapsed){
53986             this.updateBody(null, box.height);
53987         }
53988         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53989     }
53990 });
53991
53992 Roo.WestLayoutRegion = function(mgr, config){
53993     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53994     if(this.split){
53995         this.split.placement = Roo.SplitBar.LEFT;
53996         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53997         this.split.el.addClass("x-layout-split-h");
53998     }
53999     var size = config.initialSize || config.width;
54000     if(typeof size != "undefined"){
54001         this.el.setWidth(size);
54002     }
54003 };
54004 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
54005     orientation: Roo.SplitBar.HORIZONTAL,
54006     getBox : function(){
54007         if(this.collapsed){
54008             return this.collapsedEl.getBox();
54009         }
54010         var box = this.el.getBox();
54011         if(this.split){
54012             box.width += this.split.el.getWidth();
54013         }
54014         return box;
54015     },
54016     
54017     updateBox : function(box){
54018         if(this.split && !this.collapsed){
54019             var sw = this.split.el.getWidth();
54020             box.width -= sw;
54021             this.split.el.setLeft(box.x+box.width);
54022             this.split.el.setTop(box.y);
54023             this.split.el.setHeight(box.height);
54024         }
54025         if(this.collapsed){
54026             this.updateBody(null, box.height);
54027         }
54028         Roo.LayoutRegion.prototype.updateBox.call(this, box);
54029     }
54030 });
54031 /*
54032  * Based on:
54033  * Ext JS Library 1.1.1
54034  * Copyright(c) 2006-2007, Ext JS, LLC.
54035  *
54036  * Originally Released Under LGPL - original licence link has changed is not relivant.
54037  *
54038  * Fork - LGPL
54039  * <script type="text/javascript">
54040  */
54041  
54042  
54043 /*
54044  * Private internal class for reading and applying state
54045  */
54046 Roo.LayoutStateManager = function(layout){
54047      // default empty state
54048      this.state = {
54049         north: {},
54050         south: {},
54051         east: {},
54052         west: {}       
54053     };
54054 };
54055
54056 Roo.LayoutStateManager.prototype = {
54057     init : function(layout, provider){
54058         this.provider = provider;
54059         var state = provider.get(layout.id+"-layout-state");
54060         if(state){
54061             var wasUpdating = layout.isUpdating();
54062             if(!wasUpdating){
54063                 layout.beginUpdate();
54064             }
54065             for(var key in state){
54066                 if(typeof state[key] != "function"){
54067                     var rstate = state[key];
54068                     var r = layout.getRegion(key);
54069                     if(r && rstate){
54070                         if(rstate.size){
54071                             r.resizeTo(rstate.size);
54072                         }
54073                         if(rstate.collapsed == true){
54074                             r.collapse(true);
54075                         }else{
54076                             r.expand(null, true);
54077                         }
54078                     }
54079                 }
54080             }
54081             if(!wasUpdating){
54082                 layout.endUpdate();
54083             }
54084             this.state = state; 
54085         }
54086         this.layout = layout;
54087         layout.on("regionresized", this.onRegionResized, this);
54088         layout.on("regioncollapsed", this.onRegionCollapsed, this);
54089         layout.on("regionexpanded", this.onRegionExpanded, this);
54090     },
54091     
54092     storeState : function(){
54093         this.provider.set(this.layout.id+"-layout-state", this.state);
54094     },
54095     
54096     onRegionResized : function(region, newSize){
54097         this.state[region.getPosition()].size = newSize;
54098         this.storeState();
54099     },
54100     
54101     onRegionCollapsed : function(region){
54102         this.state[region.getPosition()].collapsed = true;
54103         this.storeState();
54104     },
54105     
54106     onRegionExpanded : function(region){
54107         this.state[region.getPosition()].collapsed = false;
54108         this.storeState();
54109     }
54110 };/*
54111  * Based on:
54112  * Ext JS Library 1.1.1
54113  * Copyright(c) 2006-2007, Ext JS, LLC.
54114  *
54115  * Originally Released Under LGPL - original licence link has changed is not relivant.
54116  *
54117  * Fork - LGPL
54118  * <script type="text/javascript">
54119  */
54120 /**
54121  * @class Roo.ContentPanel
54122  * @extends Roo.util.Observable
54123  * A basic ContentPanel element.
54124  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
54125  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
54126  * @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
54127  * @cfg {Boolean}   closable      True if the panel can be closed/removed
54128  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
54129  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
54130  * @cfg {Toolbar}   toolbar       A toolbar for this panel
54131  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
54132  * @cfg {String} title          The title for this panel
54133  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
54134  * @cfg {String} url            Calls {@link #setUrl} with this value
54135  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
54136  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
54137  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
54138  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
54139  * @cfg {String}    style  Extra style to add to the content panel 
54140
54141  * @constructor
54142  * Create a new ContentPanel.
54143  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
54144  * @param {String/Object} config A string to set only the title or a config object
54145  * @param {String} content (optional) Set the HTML content for this panel
54146  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
54147  */
54148 Roo.ContentPanel = function(el, config, content){
54149     
54150      
54151     /*
54152     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
54153         config = el;
54154         el = Roo.id();
54155     }
54156     if (config && config.parentLayout) { 
54157         el = config.parentLayout.el.createChild(); 
54158     }
54159     */
54160     if(el.autoCreate){ // xtype is available if this is called from factory
54161         config = el;
54162         el = Roo.id();
54163     }
54164     this.el = Roo.get(el);
54165     if(!this.el && config && config.autoCreate){
54166         if(typeof config.autoCreate == "object"){
54167             if(!config.autoCreate.id){
54168                 config.autoCreate.id = config.id||el;
54169             }
54170             this.el = Roo.DomHelper.append(document.body,
54171                         config.autoCreate, true);
54172         }else{
54173             this.el = Roo.DomHelper.append(document.body,
54174                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54175         }
54176     }
54177     
54178     
54179     this.closable = false;
54180     this.loaded = false;
54181     this.active = false;
54182     if(typeof config == "string"){
54183         this.title = config;
54184     }else{
54185         Roo.apply(this, config);
54186     }
54187     
54188     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54189         this.wrapEl = this.el.wrap();
54190         this.toolbar.container = this.el.insertSibling(false, 'before');
54191         this.toolbar = new Roo.Toolbar(this.toolbar);
54192     }
54193     
54194     // xtype created footer. - not sure if will work as we normally have to render first..
54195     if (this.footer && !this.footer.el && this.footer.xtype) {
54196         if (!this.wrapEl) {
54197             this.wrapEl = this.el.wrap();
54198         }
54199     
54200         this.footer.container = this.wrapEl.createChild();
54201          
54202         this.footer = Roo.factory(this.footer, Roo);
54203         
54204     }
54205     
54206     if(this.resizeEl){
54207         this.resizeEl = Roo.get(this.resizeEl, true);
54208     }else{
54209         this.resizeEl = this.el;
54210     }
54211     // handle view.xtype
54212     
54213  
54214     
54215     
54216     this.addEvents({
54217         /**
54218          * @event activate
54219          * Fires when this panel is activated. 
54220          * @param {Roo.ContentPanel} this
54221          */
54222         "activate" : true,
54223         /**
54224          * @event deactivate
54225          * Fires when this panel is activated. 
54226          * @param {Roo.ContentPanel} this
54227          */
54228         "deactivate" : true,
54229
54230         /**
54231          * @event resize
54232          * Fires when this panel is resized if fitToFrame is true.
54233          * @param {Roo.ContentPanel} this
54234          * @param {Number} width The width after any component adjustments
54235          * @param {Number} height The height after any component adjustments
54236          */
54237         "resize" : true,
54238         
54239          /**
54240          * @event render
54241          * Fires when this tab is created
54242          * @param {Roo.ContentPanel} this
54243          */
54244         "render" : true
54245          
54246         
54247     });
54248     
54249
54250     
54251     
54252     if(this.autoScroll){
54253         this.resizeEl.setStyle("overflow", "auto");
54254     } else {
54255         // fix randome scrolling
54256         this.el.on('scroll', function() {
54257             Roo.log('fix random scolling');
54258             this.scrollTo('top',0); 
54259         });
54260     }
54261     content = content || this.content;
54262     if(content){
54263         this.setContent(content);
54264     }
54265     if(config && config.url){
54266         this.setUrl(this.url, this.params, this.loadOnce);
54267     }
54268     
54269     
54270     
54271     Roo.ContentPanel.superclass.constructor.call(this);
54272     
54273     if (this.view && typeof(this.view.xtype) != 'undefined') {
54274         this.view.el = this.el.appendChild(document.createElement("div"));
54275         this.view = Roo.factory(this.view); 
54276         this.view.render  &&  this.view.render(false, '');  
54277     }
54278     
54279     
54280     this.fireEvent('render', this);
54281 };
54282
54283 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54284     tabTip:'',
54285     setRegion : function(region){
54286         this.region = region;
54287         if(region){
54288            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54289         }else{
54290            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54291         } 
54292     },
54293     
54294     /**
54295      * Returns the toolbar for this Panel if one was configured. 
54296      * @return {Roo.Toolbar} 
54297      */
54298     getToolbar : function(){
54299         return this.toolbar;
54300     },
54301     
54302     setActiveState : function(active){
54303         this.active = active;
54304         if(!active){
54305             this.fireEvent("deactivate", this);
54306         }else{
54307             this.fireEvent("activate", this);
54308         }
54309     },
54310     /**
54311      * Updates this panel's element
54312      * @param {String} content The new content
54313      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54314     */
54315     setContent : function(content, loadScripts){
54316         this.el.update(content, loadScripts);
54317     },
54318
54319     ignoreResize : function(w, h){
54320         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54321             return true;
54322         }else{
54323             this.lastSize = {width: w, height: h};
54324             return false;
54325         }
54326     },
54327     /**
54328      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54329      * @return {Roo.UpdateManager} The UpdateManager
54330      */
54331     getUpdateManager : function(){
54332         return this.el.getUpdateManager();
54333     },
54334      /**
54335      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54336      * @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:
54337 <pre><code>
54338 panel.load({
54339     url: "your-url.php",
54340     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54341     callback: yourFunction,
54342     scope: yourObject, //(optional scope)
54343     discardUrl: false,
54344     nocache: false,
54345     text: "Loading...",
54346     timeout: 30,
54347     scripts: false
54348 });
54349 </code></pre>
54350      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54351      * 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.
54352      * @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}
54353      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54354      * @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.
54355      * @return {Roo.ContentPanel} this
54356      */
54357     load : function(){
54358         var um = this.el.getUpdateManager();
54359         um.update.apply(um, arguments);
54360         return this;
54361     },
54362
54363
54364     /**
54365      * 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.
54366      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54367      * @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)
54368      * @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)
54369      * @return {Roo.UpdateManager} The UpdateManager
54370      */
54371     setUrl : function(url, params, loadOnce){
54372         if(this.refreshDelegate){
54373             this.removeListener("activate", this.refreshDelegate);
54374         }
54375         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54376         this.on("activate", this.refreshDelegate);
54377         return this.el.getUpdateManager();
54378     },
54379     
54380     _handleRefresh : function(url, params, loadOnce){
54381         if(!loadOnce || !this.loaded){
54382             var updater = this.el.getUpdateManager();
54383             updater.update(url, params, this._setLoaded.createDelegate(this));
54384         }
54385     },
54386     
54387     _setLoaded : function(){
54388         this.loaded = true;
54389     }, 
54390     
54391     /**
54392      * Returns this panel's id
54393      * @return {String} 
54394      */
54395     getId : function(){
54396         return this.el.id;
54397     },
54398     
54399     /** 
54400      * Returns this panel's element - used by regiosn to add.
54401      * @return {Roo.Element} 
54402      */
54403     getEl : function(){
54404         return this.wrapEl || this.el;
54405     },
54406     
54407     adjustForComponents : function(width, height)
54408     {
54409         //Roo.log('adjustForComponents ');
54410         if(this.resizeEl != this.el){
54411             width -= this.el.getFrameWidth('lr');
54412             height -= this.el.getFrameWidth('tb');
54413         }
54414         if(this.toolbar){
54415             var te = this.toolbar.getEl();
54416             height -= te.getHeight();
54417             te.setWidth(width);
54418         }
54419         if(this.footer){
54420             var te = this.footer.getEl();
54421             //Roo.log("footer:" + te.getHeight());
54422             
54423             height -= te.getHeight();
54424             te.setWidth(width);
54425         }
54426         
54427         
54428         if(this.adjustments){
54429             width += this.adjustments[0];
54430             height += this.adjustments[1];
54431         }
54432         return {"width": width, "height": height};
54433     },
54434     
54435     setSize : function(width, height){
54436         if(this.fitToFrame && !this.ignoreResize(width, height)){
54437             if(this.fitContainer && this.resizeEl != this.el){
54438                 this.el.setSize(width, height);
54439             }
54440             var size = this.adjustForComponents(width, height);
54441             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54442             this.fireEvent('resize', this, size.width, size.height);
54443         }
54444     },
54445     
54446     /**
54447      * Returns this panel's title
54448      * @return {String} 
54449      */
54450     getTitle : function(){
54451         return this.title;
54452     },
54453     
54454     /**
54455      * Set this panel's title
54456      * @param {String} title
54457      */
54458     setTitle : function(title){
54459         this.title = title;
54460         if(this.region){
54461             this.region.updatePanelTitle(this, title);
54462         }
54463     },
54464     
54465     /**
54466      * Returns true is this panel was configured to be closable
54467      * @return {Boolean} 
54468      */
54469     isClosable : function(){
54470         return this.closable;
54471     },
54472     
54473     beforeSlide : function(){
54474         this.el.clip();
54475         this.resizeEl.clip();
54476     },
54477     
54478     afterSlide : function(){
54479         this.el.unclip();
54480         this.resizeEl.unclip();
54481     },
54482     
54483     /**
54484      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54485      *   Will fail silently if the {@link #setUrl} method has not been called.
54486      *   This does not activate the panel, just updates its content.
54487      */
54488     refresh : function(){
54489         if(this.refreshDelegate){
54490            this.loaded = false;
54491            this.refreshDelegate();
54492         }
54493     },
54494     
54495     /**
54496      * Destroys this panel
54497      */
54498     destroy : function(){
54499         this.el.removeAllListeners();
54500         var tempEl = document.createElement("span");
54501         tempEl.appendChild(this.el.dom);
54502         tempEl.innerHTML = "";
54503         this.el.remove();
54504         this.el = null;
54505     },
54506     
54507     /**
54508      * form - if the content panel contains a form - this is a reference to it.
54509      * @type {Roo.form.Form}
54510      */
54511     form : false,
54512     /**
54513      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54514      *    This contains a reference to it.
54515      * @type {Roo.View}
54516      */
54517     view : false,
54518     
54519       /**
54520      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54521      * <pre><code>
54522
54523 layout.addxtype({
54524        xtype : 'Form',
54525        items: [ .... ]
54526    }
54527 );
54528
54529 </code></pre>
54530      * @param {Object} cfg Xtype definition of item to add.
54531      */
54532     
54533     addxtype : function(cfg) {
54534         // add form..
54535         if (cfg.xtype.match(/^Form$/)) {
54536             
54537             var el;
54538             //if (this.footer) {
54539             //    el = this.footer.container.insertSibling(false, 'before');
54540             //} else {
54541                 el = this.el.createChild();
54542             //}
54543
54544             this.form = new  Roo.form.Form(cfg);
54545             
54546             
54547             if ( this.form.allItems.length) {
54548                 this.form.render(el.dom);
54549             }
54550             return this.form;
54551         }
54552         // should only have one of theses..
54553         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54554             // views.. should not be just added - used named prop 'view''
54555             
54556             cfg.el = this.el.appendChild(document.createElement("div"));
54557             // factory?
54558             
54559             var ret = new Roo.factory(cfg);
54560              
54561              ret.render && ret.render(false, ''); // render blank..
54562             this.view = ret;
54563             return ret;
54564         }
54565         return false;
54566     }
54567 });
54568
54569 /**
54570  * @class Roo.GridPanel
54571  * @extends Roo.ContentPanel
54572  * @constructor
54573  * Create a new GridPanel.
54574  * @param {Roo.grid.Grid} grid The grid for this panel
54575  * @param {String/Object} config A string to set only the panel's title, or a config object
54576  */
54577 Roo.GridPanel = function(grid, config){
54578     
54579   
54580     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54581         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54582         
54583     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54584     
54585     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54586     
54587     if(this.toolbar){
54588         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54589     }
54590     // xtype created footer. - not sure if will work as we normally have to render first..
54591     if (this.footer && !this.footer.el && this.footer.xtype) {
54592         
54593         this.footer.container = this.grid.getView().getFooterPanel(true);
54594         this.footer.dataSource = this.grid.dataSource;
54595         this.footer = Roo.factory(this.footer, Roo);
54596         
54597     }
54598     
54599     grid.monitorWindowResize = false; // turn off autosizing
54600     grid.autoHeight = false;
54601     grid.autoWidth = false;
54602     this.grid = grid;
54603     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54604 };
54605
54606 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54607     getId : function(){
54608         return this.grid.id;
54609     },
54610     
54611     /**
54612      * Returns the grid for this panel
54613      * @return {Roo.grid.Grid} 
54614      */
54615     getGrid : function(){
54616         return this.grid;    
54617     },
54618     
54619     setSize : function(width, height){
54620         if(!this.ignoreResize(width, height)){
54621             var grid = this.grid;
54622             var size = this.adjustForComponents(width, height);
54623             grid.getGridEl().setSize(size.width, size.height);
54624             grid.autoSize();
54625         }
54626     },
54627     
54628     beforeSlide : function(){
54629         this.grid.getView().scroller.clip();
54630     },
54631     
54632     afterSlide : function(){
54633         this.grid.getView().scroller.unclip();
54634     },
54635     
54636     destroy : function(){
54637         this.grid.destroy();
54638         delete this.grid;
54639         Roo.GridPanel.superclass.destroy.call(this); 
54640     }
54641 });
54642
54643
54644 /**
54645  * @class Roo.NestedLayoutPanel
54646  * @extends Roo.ContentPanel
54647  * @constructor
54648  * Create a new NestedLayoutPanel.
54649  * 
54650  * 
54651  * @param {Roo.BorderLayout} layout The layout for this panel
54652  * @param {String/Object} config A string to set only the title or a config object
54653  */
54654 Roo.NestedLayoutPanel = function(layout, config)
54655 {
54656     // construct with only one argument..
54657     /* FIXME - implement nicer consturctors
54658     if (layout.layout) {
54659         config = layout;
54660         layout = config.layout;
54661         delete config.layout;
54662     }
54663     if (layout.xtype && !layout.getEl) {
54664         // then layout needs constructing..
54665         layout = Roo.factory(layout, Roo);
54666     }
54667     */
54668     
54669     
54670     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54671     
54672     layout.monitorWindowResize = false; // turn off autosizing
54673     this.layout = layout;
54674     this.layout.getEl().addClass("x-layout-nested-layout");
54675     
54676     
54677     
54678     
54679 };
54680
54681 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54682
54683     setSize : function(width, height){
54684         if(!this.ignoreResize(width, height)){
54685             var size = this.adjustForComponents(width, height);
54686             var el = this.layout.getEl();
54687             el.setSize(size.width, size.height);
54688             var touch = el.dom.offsetWidth;
54689             this.layout.layout();
54690             // ie requires a double layout on the first pass
54691             if(Roo.isIE && !this.initialized){
54692                 this.initialized = true;
54693                 this.layout.layout();
54694             }
54695         }
54696     },
54697     
54698     // activate all subpanels if not currently active..
54699     
54700     setActiveState : function(active){
54701         this.active = active;
54702         if(!active){
54703             this.fireEvent("deactivate", this);
54704             return;
54705         }
54706         
54707         this.fireEvent("activate", this);
54708         // not sure if this should happen before or after..
54709         if (!this.layout) {
54710             return; // should not happen..
54711         }
54712         var reg = false;
54713         for (var r in this.layout.regions) {
54714             reg = this.layout.getRegion(r);
54715             if (reg.getActivePanel()) {
54716                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54717                 reg.setActivePanel(reg.getActivePanel());
54718                 continue;
54719             }
54720             if (!reg.panels.length) {
54721                 continue;
54722             }
54723             reg.showPanel(reg.getPanel(0));
54724         }
54725         
54726         
54727         
54728         
54729     },
54730     
54731     /**
54732      * Returns the nested BorderLayout for this panel
54733      * @return {Roo.BorderLayout} 
54734      */
54735     getLayout : function(){
54736         return this.layout;
54737     },
54738     
54739      /**
54740      * Adds a xtype elements to the layout of the nested panel
54741      * <pre><code>
54742
54743 panel.addxtype({
54744        xtype : 'ContentPanel',
54745        region: 'west',
54746        items: [ .... ]
54747    }
54748 );
54749
54750 panel.addxtype({
54751         xtype : 'NestedLayoutPanel',
54752         region: 'west',
54753         layout: {
54754            center: { },
54755            west: { }   
54756         },
54757         items : [ ... list of content panels or nested layout panels.. ]
54758    }
54759 );
54760 </code></pre>
54761      * @param {Object} cfg Xtype definition of item to add.
54762      */
54763     addxtype : function(cfg) {
54764         return this.layout.addxtype(cfg);
54765     
54766     }
54767 });
54768
54769 Roo.ScrollPanel = function(el, config, content){
54770     config = config || {};
54771     config.fitToFrame = true;
54772     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54773     
54774     this.el.dom.style.overflow = "hidden";
54775     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54776     this.el.removeClass("x-layout-inactive-content");
54777     this.el.on("mousewheel", this.onWheel, this);
54778
54779     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54780     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54781     up.unselectable(); down.unselectable();
54782     up.on("click", this.scrollUp, this);
54783     down.on("click", this.scrollDown, this);
54784     up.addClassOnOver("x-scroller-btn-over");
54785     down.addClassOnOver("x-scroller-btn-over");
54786     up.addClassOnClick("x-scroller-btn-click");
54787     down.addClassOnClick("x-scroller-btn-click");
54788     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54789
54790     this.resizeEl = this.el;
54791     this.el = wrap; this.up = up; this.down = down;
54792 };
54793
54794 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54795     increment : 100,
54796     wheelIncrement : 5,
54797     scrollUp : function(){
54798         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54799     },
54800
54801     scrollDown : function(){
54802         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54803     },
54804
54805     afterScroll : function(){
54806         var el = this.resizeEl;
54807         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54808         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54809         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54810     },
54811
54812     setSize : function(){
54813         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54814         this.afterScroll();
54815     },
54816
54817     onWheel : function(e){
54818         var d = e.getWheelDelta();
54819         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54820         this.afterScroll();
54821         e.stopEvent();
54822     },
54823
54824     setContent : function(content, loadScripts){
54825         this.resizeEl.update(content, loadScripts);
54826     }
54827
54828 });
54829
54830
54831
54832
54833
54834
54835
54836
54837
54838 /**
54839  * @class Roo.TreePanel
54840  * @extends Roo.ContentPanel
54841  * @constructor
54842  * Create a new TreePanel. - defaults to fit/scoll contents.
54843  * @param {String/Object} config A string to set only the panel's title, or a config object
54844  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54845  */
54846 Roo.TreePanel = function(config){
54847     var el = config.el;
54848     var tree = config.tree;
54849     delete config.tree; 
54850     delete config.el; // hopefull!
54851     
54852     // wrapper for IE7 strict & safari scroll issue
54853     
54854     var treeEl = el.createChild();
54855     config.resizeEl = treeEl;
54856     
54857     
54858     
54859     Roo.TreePanel.superclass.constructor.call(this, el, config);
54860  
54861  
54862     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54863     //console.log(tree);
54864     this.on('activate', function()
54865     {
54866         if (this.tree.rendered) {
54867             return;
54868         }
54869         //console.log('render tree');
54870         this.tree.render();
54871     });
54872     // this should not be needed.. - it's actually the 'el' that resizes?
54873     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54874     
54875     //this.on('resize',  function (cp, w, h) {
54876     //        this.tree.innerCt.setWidth(w);
54877     //        this.tree.innerCt.setHeight(h);
54878     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54879     //});
54880
54881         
54882     
54883 };
54884
54885 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54886     fitToFrame : true,
54887     autoScroll : true
54888 });
54889
54890
54891
54892
54893
54894
54895
54896
54897
54898
54899
54900 /*
54901  * Based on:
54902  * Ext JS Library 1.1.1
54903  * Copyright(c) 2006-2007, Ext JS, LLC.
54904  *
54905  * Originally Released Under LGPL - original licence link has changed is not relivant.
54906  *
54907  * Fork - LGPL
54908  * <script type="text/javascript">
54909  */
54910  
54911
54912 /**
54913  * @class Roo.ReaderLayout
54914  * @extends Roo.BorderLayout
54915  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54916  * center region containing two nested regions (a top one for a list view and one for item preview below),
54917  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54918  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54919  * expedites the setup of the overall layout and regions for this common application style.
54920  * Example:
54921  <pre><code>
54922 var reader = new Roo.ReaderLayout();
54923 var CP = Roo.ContentPanel;  // shortcut for adding
54924
54925 reader.beginUpdate();
54926 reader.add("north", new CP("north", "North"));
54927 reader.add("west", new CP("west", {title: "West"}));
54928 reader.add("east", new CP("east", {title: "East"}));
54929
54930 reader.regions.listView.add(new CP("listView", "List"));
54931 reader.regions.preview.add(new CP("preview", "Preview"));
54932 reader.endUpdate();
54933 </code></pre>
54934 * @constructor
54935 * Create a new ReaderLayout
54936 * @param {Object} config Configuration options
54937 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54938 * document.body if omitted)
54939 */
54940 Roo.ReaderLayout = function(config, renderTo){
54941     var c = config || {size:{}};
54942     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54943         north: c.north !== false ? Roo.apply({
54944             split:false,
54945             initialSize: 32,
54946             titlebar: false
54947         }, c.north) : false,
54948         west: c.west !== false ? Roo.apply({
54949             split:true,
54950             initialSize: 200,
54951             minSize: 175,
54952             maxSize: 400,
54953             titlebar: true,
54954             collapsible: true,
54955             animate: true,
54956             margins:{left:5,right:0,bottom:5,top:5},
54957             cmargins:{left:5,right:5,bottom:5,top:5}
54958         }, c.west) : false,
54959         east: c.east !== false ? Roo.apply({
54960             split:true,
54961             initialSize: 200,
54962             minSize: 175,
54963             maxSize: 400,
54964             titlebar: true,
54965             collapsible: true,
54966             animate: true,
54967             margins:{left:0,right:5,bottom:5,top:5},
54968             cmargins:{left:5,right:5,bottom:5,top:5}
54969         }, c.east) : false,
54970         center: Roo.apply({
54971             tabPosition: 'top',
54972             autoScroll:false,
54973             closeOnTab: true,
54974             titlebar:false,
54975             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54976         }, c.center)
54977     });
54978
54979     this.el.addClass('x-reader');
54980
54981     this.beginUpdate();
54982
54983     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54984         south: c.preview !== false ? Roo.apply({
54985             split:true,
54986             initialSize: 200,
54987             minSize: 100,
54988             autoScroll:true,
54989             collapsible:true,
54990             titlebar: true,
54991             cmargins:{top:5,left:0, right:0, bottom:0}
54992         }, c.preview) : false,
54993         center: Roo.apply({
54994             autoScroll:false,
54995             titlebar:false,
54996             minHeight:200
54997         }, c.listView)
54998     });
54999     this.add('center', new Roo.NestedLayoutPanel(inner,
55000             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
55001
55002     this.endUpdate();
55003
55004     this.regions.preview = inner.getRegion('south');
55005     this.regions.listView = inner.getRegion('center');
55006 };
55007
55008 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
55009  * Based on:
55010  * Ext JS Library 1.1.1
55011  * Copyright(c) 2006-2007, Ext JS, LLC.
55012  *
55013  * Originally Released Under LGPL - original licence link has changed is not relivant.
55014  *
55015  * Fork - LGPL
55016  * <script type="text/javascript">
55017  */
55018  
55019 /**
55020  * @class Roo.grid.Grid
55021  * @extends Roo.util.Observable
55022  * This class represents the primary interface of a component based grid control.
55023  * <br><br>Usage:<pre><code>
55024  var grid = new Roo.grid.Grid("my-container-id", {
55025      ds: myDataStore,
55026      cm: myColModel,
55027      selModel: mySelectionModel,
55028      autoSizeColumns: true,
55029      monitorWindowResize: false,
55030      trackMouseOver: true
55031  });
55032  // set any options
55033  grid.render();
55034  * </code></pre>
55035  * <b>Common Problems:</b><br/>
55036  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
55037  * element will correct this<br/>
55038  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
55039  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
55040  * are unpredictable.<br/>
55041  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
55042  * grid to calculate dimensions/offsets.<br/>
55043   * @constructor
55044  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
55045  * The container MUST have some type of size defined for the grid to fill. The container will be
55046  * automatically set to position relative if it isn't already.
55047  * @param {Object} config A config object that sets properties on this grid.
55048  */
55049 Roo.grid.Grid = function(container, config){
55050         // initialize the container
55051         this.container = Roo.get(container);
55052         this.container.update("");
55053         this.container.setStyle("overflow", "hidden");
55054     this.container.addClass('x-grid-container');
55055
55056     this.id = this.container.id;
55057
55058     Roo.apply(this, config);
55059     // check and correct shorthanded configs
55060     if(this.ds){
55061         this.dataSource = this.ds;
55062         delete this.ds;
55063     }
55064     if(this.cm){
55065         this.colModel = this.cm;
55066         delete this.cm;
55067     }
55068     if(this.sm){
55069         this.selModel = this.sm;
55070         delete this.sm;
55071     }
55072
55073     if (this.selModel) {
55074         this.selModel = Roo.factory(this.selModel, Roo.grid);
55075         this.sm = this.selModel;
55076         this.sm.xmodule = this.xmodule || false;
55077     }
55078     if (typeof(this.colModel.config) == 'undefined') {
55079         this.colModel = new Roo.grid.ColumnModel(this.colModel);
55080         this.cm = this.colModel;
55081         this.cm.xmodule = this.xmodule || false;
55082     }
55083     if (this.dataSource) {
55084         this.dataSource= Roo.factory(this.dataSource, Roo.data);
55085         this.ds = this.dataSource;
55086         this.ds.xmodule = this.xmodule || false;
55087          
55088     }
55089     
55090     
55091     
55092     if(this.width){
55093         this.container.setWidth(this.width);
55094     }
55095
55096     if(this.height){
55097         this.container.setHeight(this.height);
55098     }
55099     /** @private */
55100         this.addEvents({
55101         // raw events
55102         /**
55103          * @event click
55104          * The raw click event for the entire grid.
55105          * @param {Roo.EventObject} e
55106          */
55107         "click" : true,
55108         /**
55109          * @event dblclick
55110          * The raw dblclick event for the entire grid.
55111          * @param {Roo.EventObject} e
55112          */
55113         "dblclick" : true,
55114         /**
55115          * @event contextmenu
55116          * The raw contextmenu event for the entire grid.
55117          * @param {Roo.EventObject} e
55118          */
55119         "contextmenu" : true,
55120         /**
55121          * @event mousedown
55122          * The raw mousedown event for the entire grid.
55123          * @param {Roo.EventObject} e
55124          */
55125         "mousedown" : true,
55126         /**
55127          * @event mouseup
55128          * The raw mouseup event for the entire grid.
55129          * @param {Roo.EventObject} e
55130          */
55131         "mouseup" : true,
55132         /**
55133          * @event mouseover
55134          * The raw mouseover event for the entire grid.
55135          * @param {Roo.EventObject} e
55136          */
55137         "mouseover" : true,
55138         /**
55139          * @event mouseout
55140          * The raw mouseout event for the entire grid.
55141          * @param {Roo.EventObject} e
55142          */
55143         "mouseout" : true,
55144         /**
55145          * @event keypress
55146          * The raw keypress event for the entire grid.
55147          * @param {Roo.EventObject} e
55148          */
55149         "keypress" : true,
55150         /**
55151          * @event keydown
55152          * The raw keydown event for the entire grid.
55153          * @param {Roo.EventObject} e
55154          */
55155         "keydown" : true,
55156
55157         // custom events
55158
55159         /**
55160          * @event cellclick
55161          * Fires when a cell is clicked
55162          * @param {Grid} this
55163          * @param {Number} rowIndex
55164          * @param {Number} columnIndex
55165          * @param {Roo.EventObject} e
55166          */
55167         "cellclick" : true,
55168         /**
55169          * @event celldblclick
55170          * Fires when a cell is double clicked
55171          * @param {Grid} this
55172          * @param {Number} rowIndex
55173          * @param {Number} columnIndex
55174          * @param {Roo.EventObject} e
55175          */
55176         "celldblclick" : true,
55177         /**
55178          * @event rowclick
55179          * Fires when a row is clicked
55180          * @param {Grid} this
55181          * @param {Number} rowIndex
55182          * @param {Roo.EventObject} e
55183          */
55184         "rowclick" : true,
55185         /**
55186          * @event rowdblclick
55187          * Fires when a row is double clicked
55188          * @param {Grid} this
55189          * @param {Number} rowIndex
55190          * @param {Roo.EventObject} e
55191          */
55192         "rowdblclick" : true,
55193         /**
55194          * @event headerclick
55195          * Fires when a header is clicked
55196          * @param {Grid} this
55197          * @param {Number} columnIndex
55198          * @param {Roo.EventObject} e
55199          */
55200         "headerclick" : true,
55201         /**
55202          * @event headerdblclick
55203          * Fires when a header cell is double clicked
55204          * @param {Grid} this
55205          * @param {Number} columnIndex
55206          * @param {Roo.EventObject} e
55207          */
55208         "headerdblclick" : true,
55209         /**
55210          * @event rowcontextmenu
55211          * Fires when a row is right clicked
55212          * @param {Grid} this
55213          * @param {Number} rowIndex
55214          * @param {Roo.EventObject} e
55215          */
55216         "rowcontextmenu" : true,
55217         /**
55218          * @event cellcontextmenu
55219          * Fires when a cell is right clicked
55220          * @param {Grid} this
55221          * @param {Number} rowIndex
55222          * @param {Number} cellIndex
55223          * @param {Roo.EventObject} e
55224          */
55225          "cellcontextmenu" : true,
55226         /**
55227          * @event headercontextmenu
55228          * Fires when a header is right clicked
55229          * @param {Grid} this
55230          * @param {Number} columnIndex
55231          * @param {Roo.EventObject} e
55232          */
55233         "headercontextmenu" : true,
55234         /**
55235          * @event bodyscroll
55236          * Fires when the body element is scrolled
55237          * @param {Number} scrollLeft
55238          * @param {Number} scrollTop
55239          */
55240         "bodyscroll" : true,
55241         /**
55242          * @event columnresize
55243          * Fires when the user resizes a column
55244          * @param {Number} columnIndex
55245          * @param {Number} newSize
55246          */
55247         "columnresize" : true,
55248         /**
55249          * @event columnmove
55250          * Fires when the user moves a column
55251          * @param {Number} oldIndex
55252          * @param {Number} newIndex
55253          */
55254         "columnmove" : true,
55255         /**
55256          * @event startdrag
55257          * Fires when row(s) start being dragged
55258          * @param {Grid} this
55259          * @param {Roo.GridDD} dd The drag drop object
55260          * @param {event} e The raw browser event
55261          */
55262         "startdrag" : true,
55263         /**
55264          * @event enddrag
55265          * Fires when a drag operation is complete
55266          * @param {Grid} this
55267          * @param {Roo.GridDD} dd The drag drop object
55268          * @param {event} e The raw browser event
55269          */
55270         "enddrag" : true,
55271         /**
55272          * @event dragdrop
55273          * Fires when dragged row(s) are dropped on a valid DD target
55274          * @param {Grid} this
55275          * @param {Roo.GridDD} dd The drag drop object
55276          * @param {String} targetId The target drag drop object
55277          * @param {event} e The raw browser event
55278          */
55279         "dragdrop" : true,
55280         /**
55281          * @event dragover
55282          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55283          * @param {Grid} this
55284          * @param {Roo.GridDD} dd The drag drop object
55285          * @param {String} targetId The target drag drop object
55286          * @param {event} e The raw browser event
55287          */
55288         "dragover" : true,
55289         /**
55290          * @event dragenter
55291          *  Fires when the dragged row(s) first cross another DD target while being dragged
55292          * @param {Grid} this
55293          * @param {Roo.GridDD} dd The drag drop object
55294          * @param {String} targetId The target drag drop object
55295          * @param {event} e The raw browser event
55296          */
55297         "dragenter" : true,
55298         /**
55299          * @event dragout
55300          * Fires when the dragged row(s) leave another DD target while being dragged
55301          * @param {Grid} this
55302          * @param {Roo.GridDD} dd The drag drop object
55303          * @param {String} targetId The target drag drop object
55304          * @param {event} e The raw browser event
55305          */
55306         "dragout" : true,
55307         /**
55308          * @event rowclass
55309          * Fires when a row is rendered, so you can change add a style to it.
55310          * @param {GridView} gridview   The grid view
55311          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55312          */
55313         'rowclass' : true,
55314
55315         /**
55316          * @event render
55317          * Fires when the grid is rendered
55318          * @param {Grid} grid
55319          */
55320         'render' : true
55321     });
55322
55323     Roo.grid.Grid.superclass.constructor.call(this);
55324 };
55325 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55326     
55327     /**
55328      * @cfg {String} ddGroup - drag drop group.
55329      */
55330
55331     /**
55332      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55333      */
55334     minColumnWidth : 25,
55335
55336     /**
55337      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55338      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55339      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55340      */
55341     autoSizeColumns : false,
55342
55343     /**
55344      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55345      */
55346     autoSizeHeaders : true,
55347
55348     /**
55349      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55350      */
55351     monitorWindowResize : true,
55352
55353     /**
55354      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55355      * rows measured to get a columns size. Default is 0 (all rows).
55356      */
55357     maxRowsToMeasure : 0,
55358
55359     /**
55360      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55361      */
55362     trackMouseOver : true,
55363
55364     /**
55365     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55366     */
55367     
55368     /**
55369     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55370     */
55371     enableDragDrop : false,
55372     
55373     /**
55374     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55375     */
55376     enableColumnMove : true,
55377     
55378     /**
55379     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55380     */
55381     enableColumnHide : true,
55382     
55383     /**
55384     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55385     */
55386     enableRowHeightSync : false,
55387     
55388     /**
55389     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55390     */
55391     stripeRows : true,
55392     
55393     /**
55394     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55395     */
55396     autoHeight : false,
55397
55398     /**
55399      * @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.
55400      */
55401     autoExpandColumn : false,
55402
55403     /**
55404     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55405     * Default is 50.
55406     */
55407     autoExpandMin : 50,
55408
55409     /**
55410     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55411     */
55412     autoExpandMax : 1000,
55413
55414     /**
55415     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55416     */
55417     view : null,
55418
55419     /**
55420     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55421     */
55422     loadMask : false,
55423     /**
55424     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55425     */
55426     dropTarget: false,
55427     
55428    
55429     
55430     // private
55431     rendered : false,
55432
55433     /**
55434     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55435     * of a fixed width. Default is false.
55436     */
55437     /**
55438     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55439     */
55440     /**
55441      * Called once after all setup has been completed and the grid is ready to be rendered.
55442      * @return {Roo.grid.Grid} this
55443      */
55444     render : function()
55445     {
55446         var c = this.container;
55447         // try to detect autoHeight/width mode
55448         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55449             this.autoHeight = true;
55450         }
55451         var view = this.getView();
55452         view.init(this);
55453
55454         c.on("click", this.onClick, this);
55455         c.on("dblclick", this.onDblClick, this);
55456         c.on("contextmenu", this.onContextMenu, this);
55457         c.on("keydown", this.onKeyDown, this);
55458         if (Roo.isTouch) {
55459             c.on("touchstart", this.onTouchStart, this);
55460         }
55461
55462         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55463
55464         this.getSelectionModel().init(this);
55465
55466         view.render();
55467
55468         if(this.loadMask){
55469             this.loadMask = new Roo.LoadMask(this.container,
55470                     Roo.apply({store:this.dataSource}, this.loadMask));
55471         }
55472         
55473         
55474         if (this.toolbar && this.toolbar.xtype) {
55475             this.toolbar.container = this.getView().getHeaderPanel(true);
55476             this.toolbar = new Roo.Toolbar(this.toolbar);
55477         }
55478         if (this.footer && this.footer.xtype) {
55479             this.footer.dataSource = this.getDataSource();
55480             this.footer.container = this.getView().getFooterPanel(true);
55481             this.footer = Roo.factory(this.footer, Roo);
55482         }
55483         if (this.dropTarget && this.dropTarget.xtype) {
55484             delete this.dropTarget.xtype;
55485             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55486         }
55487         
55488         
55489         this.rendered = true;
55490         this.fireEvent('render', this);
55491         return this;
55492     },
55493
55494     /**
55495      * Reconfigures the grid to use a different Store and Column Model.
55496      * The View will be bound to the new objects and refreshed.
55497      * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55498      * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55499      */
55500     reconfigure : function(dataSource, colModel){
55501         if(this.loadMask){
55502             this.loadMask.destroy();
55503             this.loadMask = new Roo.LoadMask(this.container,
55504                     Roo.apply({store:dataSource}, this.loadMask));
55505         }
55506         this.view.bind(dataSource, colModel);
55507         this.dataSource = dataSource;
55508         this.colModel = colModel;
55509         this.view.refresh(true);
55510     },
55511     /**
55512      * addColumns
55513      * Add's a column, default at the end..
55514      
55515      * @param {int} position to add (default end)
55516      * @param {Array} of objects of column configuration see {@link Roo.grid.ColumnModel} 
55517      */
55518     addColumns : function(pos, ar)
55519     {
55520         
55521         for (var i =0;i< ar.length;i++) {
55522             var cfg = ar[i];
55523             cfg.id = typeof(cfg.id) == 'undefined' ? Roo.id() : cfg.id; // don't normally use this..
55524             this.cm.lookup[cfg.id] = cfg;
55525         }
55526         
55527         
55528         if (typeof(pos) == 'undefined' || pos >= this.cm.config.length) {
55529             pos = this.cm.config.length; //this.cm.config.push(cfg);
55530         } 
55531         pos = Math.max(0,pos);
55532         ar.unshift(0);
55533         ar.unshift(pos);
55534         this.cm.config.splice.apply(this.cm.config, ar);
55535         
55536         
55537         
55538         this.view.generateRules(this.cm);
55539         this.view.refresh(true);
55540         
55541     },
55542     
55543     
55544     
55545     
55546     // private
55547     onKeyDown : function(e){
55548         this.fireEvent("keydown", e);
55549     },
55550
55551     /**
55552      * Destroy this grid.
55553      * @param {Boolean} removeEl True to remove the element
55554      */
55555     destroy : function(removeEl, keepListeners){
55556         if(this.loadMask){
55557             this.loadMask.destroy();
55558         }
55559         var c = this.container;
55560         c.removeAllListeners();
55561         this.view.destroy();
55562         this.colModel.purgeListeners();
55563         if(!keepListeners){
55564             this.purgeListeners();
55565         }
55566         c.update("");
55567         if(removeEl === true){
55568             c.remove();
55569         }
55570     },
55571
55572     // private
55573     processEvent : function(name, e){
55574         // does this fire select???
55575         //Roo.log('grid:processEvent '  + name);
55576         
55577         if (name != 'touchstart' ) {
55578             this.fireEvent(name, e);    
55579         }
55580         
55581         var t = e.getTarget();
55582         var v = this.view;
55583         var header = v.findHeaderIndex(t);
55584         if(header !== false){
55585             var ename = name == 'touchstart' ? 'click' : name;
55586              
55587             this.fireEvent("header" + ename, this, header, e);
55588         }else{
55589             var row = v.findRowIndex(t);
55590             var cell = v.findCellIndex(t);
55591             if (name == 'touchstart') {
55592                 // first touch is always a click.
55593                 // hopefull this happens after selection is updated.?
55594                 name = false;
55595                 
55596                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55597                     var cs = this.selModel.getSelectedCell();
55598                     if (row == cs[0] && cell == cs[1]){
55599                         name = 'dblclick';
55600                     }
55601                 }
55602                 if (typeof(this.selModel.getSelections) != 'undefined') {
55603                     var cs = this.selModel.getSelections();
55604                     var ds = this.dataSource;
55605                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55606                         name = 'dblclick';
55607                     }
55608                 }
55609                 if (!name) {
55610                     return;
55611                 }
55612             }
55613             
55614             
55615             if(row !== false){
55616                 this.fireEvent("row" + name, this, row, e);
55617                 if(cell !== false){
55618                     this.fireEvent("cell" + name, this, row, cell, e);
55619                 }
55620             }
55621         }
55622     },
55623
55624     // private
55625     onClick : function(e){
55626         this.processEvent("click", e);
55627     },
55628    // private
55629     onTouchStart : function(e){
55630         this.processEvent("touchstart", e);
55631     },
55632
55633     // private
55634     onContextMenu : function(e, t){
55635         this.processEvent("contextmenu", e);
55636     },
55637
55638     // private
55639     onDblClick : function(e){
55640         this.processEvent("dblclick", e);
55641     },
55642
55643     // private
55644     walkCells : function(row, col, step, fn, scope){
55645         var cm = this.colModel, clen = cm.getColumnCount();
55646         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55647         if(step < 0){
55648             if(col < 0){
55649                 row--;
55650                 first = false;
55651             }
55652             while(row >= 0){
55653                 if(!first){
55654                     col = clen-1;
55655                 }
55656                 first = false;
55657                 while(col >= 0){
55658                     if(fn.call(scope || this, row, col, cm) === true){
55659                         return [row, col];
55660                     }
55661                     col--;
55662                 }
55663                 row--;
55664             }
55665         } else {
55666             if(col >= clen){
55667                 row++;
55668                 first = false;
55669             }
55670             while(row < rlen){
55671                 if(!first){
55672                     col = 0;
55673                 }
55674                 first = false;
55675                 while(col < clen){
55676                     if(fn.call(scope || this, row, col, cm) === true){
55677                         return [row, col];
55678                     }
55679                     col++;
55680                 }
55681                 row++;
55682             }
55683         }
55684         return null;
55685     },
55686
55687     // private
55688     getSelections : function(){
55689         return this.selModel.getSelections();
55690     },
55691
55692     /**
55693      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55694      * but if manual update is required this method will initiate it.
55695      */
55696     autoSize : function(){
55697         if(this.rendered){
55698             this.view.layout();
55699             if(this.view.adjustForScroll){
55700                 this.view.adjustForScroll();
55701             }
55702         }
55703     },
55704
55705     /**
55706      * Returns the grid's underlying element.
55707      * @return {Element} The element
55708      */
55709     getGridEl : function(){
55710         return this.container;
55711     },
55712
55713     // private for compatibility, overridden by editor grid
55714     stopEditing : function(){},
55715
55716     /**
55717      * Returns the grid's SelectionModel.
55718      * @return {SelectionModel}
55719      */
55720     getSelectionModel : function(){
55721         if(!this.selModel){
55722             this.selModel = new Roo.grid.RowSelectionModel();
55723         }
55724         return this.selModel;
55725     },
55726
55727     /**
55728      * Returns the grid's DataSource.
55729      * @return {DataSource}
55730      */
55731     getDataSource : function(){
55732         return this.dataSource;
55733     },
55734
55735     /**
55736      * Returns the grid's ColumnModel.
55737      * @return {ColumnModel}
55738      */
55739     getColumnModel : function(){
55740         return this.colModel;
55741     },
55742
55743     /**
55744      * Returns the grid's GridView object.
55745      * @return {GridView}
55746      */
55747     getView : function(){
55748         if(!this.view){
55749             this.view = new Roo.grid.GridView(this.viewConfig);
55750         }
55751         return this.view;
55752     },
55753     /**
55754      * Called to get grid's drag proxy text, by default returns this.ddText.
55755      * @return {String}
55756      */
55757     getDragDropText : function(){
55758         var count = this.selModel.getCount();
55759         return String.format(this.ddText, count, count == 1 ? '' : 's');
55760     }
55761 });
55762 /**
55763  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55764  * %0 is replaced with the number of selected rows.
55765  * @type String
55766  */
55767 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55768  * Based on:
55769  * Ext JS Library 1.1.1
55770  * Copyright(c) 2006-2007, Ext JS, LLC.
55771  *
55772  * Originally Released Under LGPL - original licence link has changed is not relivant.
55773  *
55774  * Fork - LGPL
55775  * <script type="text/javascript">
55776  */
55777  
55778 Roo.grid.AbstractGridView = function(){
55779         this.grid = null;
55780         
55781         this.events = {
55782             "beforerowremoved" : true,
55783             "beforerowsinserted" : true,
55784             "beforerefresh" : true,
55785             "rowremoved" : true,
55786             "rowsinserted" : true,
55787             "rowupdated" : true,
55788             "refresh" : true
55789         };
55790     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55791 };
55792
55793 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55794     rowClass : "x-grid-row",
55795     cellClass : "x-grid-cell",
55796     tdClass : "x-grid-td",
55797     hdClass : "x-grid-hd",
55798     splitClass : "x-grid-hd-split",
55799     
55800     init: function(grid){
55801         this.grid = grid;
55802                 var cid = this.grid.getGridEl().id;
55803         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55804         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55805         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55806         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55807         },
55808         
55809     getColumnRenderers : function(){
55810         var renderers = [];
55811         var cm = this.grid.colModel;
55812         var colCount = cm.getColumnCount();
55813         for(var i = 0; i < colCount; i++){
55814             renderers[i] = cm.getRenderer(i);
55815         }
55816         return renderers;
55817     },
55818     
55819     getColumnIds : function(){
55820         var ids = [];
55821         var cm = this.grid.colModel;
55822         var colCount = cm.getColumnCount();
55823         for(var i = 0; i < colCount; i++){
55824             ids[i] = cm.getColumnId(i);
55825         }
55826         return ids;
55827     },
55828     
55829     getDataIndexes : function(){
55830         if(!this.indexMap){
55831             this.indexMap = this.buildIndexMap();
55832         }
55833         return this.indexMap.colToData;
55834     },
55835     
55836     getColumnIndexByDataIndex : function(dataIndex){
55837         if(!this.indexMap){
55838             this.indexMap = this.buildIndexMap();
55839         }
55840         return this.indexMap.dataToCol[dataIndex];
55841     },
55842     
55843     /**
55844      * Set a css style for a column dynamically. 
55845      * @param {Number} colIndex The index of the column
55846      * @param {String} name The css property name
55847      * @param {String} value The css value
55848      */
55849     setCSSStyle : function(colIndex, name, value){
55850         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55851         Roo.util.CSS.updateRule(selector, name, value);
55852     },
55853     
55854     generateRules : function(cm){
55855         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55856         Roo.util.CSS.removeStyleSheet(rulesId);
55857         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55858             var cid = cm.getColumnId(i);
55859             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55860                          this.tdSelector, cid, " {\n}\n",
55861                          this.hdSelector, cid, " {\n}\n",
55862                          this.splitSelector, cid, " {\n}\n");
55863         }
55864         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55865     }
55866 });/*
55867  * Based on:
55868  * Ext JS Library 1.1.1
55869  * Copyright(c) 2006-2007, Ext JS, LLC.
55870  *
55871  * Originally Released Under LGPL - original licence link has changed is not relivant.
55872  *
55873  * Fork - LGPL
55874  * <script type="text/javascript">
55875  */
55876
55877 // private
55878 // This is a support class used internally by the Grid components
55879 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55880     this.grid = grid;
55881     this.view = grid.getView();
55882     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55883     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55884     if(hd2){
55885         this.setHandleElId(Roo.id(hd));
55886         this.setOuterHandleElId(Roo.id(hd2));
55887     }
55888     this.scroll = false;
55889 };
55890 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55891     maxDragWidth: 120,
55892     getDragData : function(e){
55893         var t = Roo.lib.Event.getTarget(e);
55894         var h = this.view.findHeaderCell(t);
55895         if(h){
55896             return {ddel: h.firstChild, header:h};
55897         }
55898         return false;
55899     },
55900
55901     onInitDrag : function(e){
55902         this.view.headersDisabled = true;
55903         var clone = this.dragData.ddel.cloneNode(true);
55904         clone.id = Roo.id();
55905         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55906         this.proxy.update(clone);
55907         return true;
55908     },
55909
55910     afterValidDrop : function(){
55911         var v = this.view;
55912         setTimeout(function(){
55913             v.headersDisabled = false;
55914         }, 50);
55915     },
55916
55917     afterInvalidDrop : function(){
55918         var v = this.view;
55919         setTimeout(function(){
55920             v.headersDisabled = false;
55921         }, 50);
55922     }
55923 });
55924 /*
55925  * Based on:
55926  * Ext JS Library 1.1.1
55927  * Copyright(c) 2006-2007, Ext JS, LLC.
55928  *
55929  * Originally Released Under LGPL - original licence link has changed is not relivant.
55930  *
55931  * Fork - LGPL
55932  * <script type="text/javascript">
55933  */
55934 // private
55935 // This is a support class used internally by the Grid components
55936 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55937     this.grid = grid;
55938     this.view = grid.getView();
55939     // split the proxies so they don't interfere with mouse events
55940     this.proxyTop = Roo.DomHelper.append(document.body, {
55941         cls:"col-move-top", html:"&#160;"
55942     }, true);
55943     this.proxyBottom = Roo.DomHelper.append(document.body, {
55944         cls:"col-move-bottom", html:"&#160;"
55945     }, true);
55946     this.proxyTop.hide = this.proxyBottom.hide = function(){
55947         this.setLeftTop(-100,-100);
55948         this.setStyle("visibility", "hidden");
55949     };
55950     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55951     // temporarily disabled
55952     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55953     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55954 };
55955 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55956     proxyOffsets : [-4, -9],
55957     fly: Roo.Element.fly,
55958
55959     getTargetFromEvent : function(e){
55960         var t = Roo.lib.Event.getTarget(e);
55961         var cindex = this.view.findCellIndex(t);
55962         if(cindex !== false){
55963             return this.view.getHeaderCell(cindex);
55964         }
55965         return null;
55966     },
55967
55968     nextVisible : function(h){
55969         var v = this.view, cm = this.grid.colModel;
55970         h = h.nextSibling;
55971         while(h){
55972             if(!cm.isHidden(v.getCellIndex(h))){
55973                 return h;
55974             }
55975             h = h.nextSibling;
55976         }
55977         return null;
55978     },
55979
55980     prevVisible : function(h){
55981         var v = this.view, cm = this.grid.colModel;
55982         h = h.prevSibling;
55983         while(h){
55984             if(!cm.isHidden(v.getCellIndex(h))){
55985                 return h;
55986             }
55987             h = h.prevSibling;
55988         }
55989         return null;
55990     },
55991
55992     positionIndicator : function(h, n, e){
55993         var x = Roo.lib.Event.getPageX(e);
55994         var r = Roo.lib.Dom.getRegion(n.firstChild);
55995         var px, pt, py = r.top + this.proxyOffsets[1];
55996         if((r.right - x) <= (r.right-r.left)/2){
55997             px = r.right+this.view.borderWidth;
55998             pt = "after";
55999         }else{
56000             px = r.left;
56001             pt = "before";
56002         }
56003         var oldIndex = this.view.getCellIndex(h);
56004         var newIndex = this.view.getCellIndex(n);
56005
56006         if(this.grid.colModel.isFixed(newIndex)){
56007             return false;
56008         }
56009
56010         var locked = this.grid.colModel.isLocked(newIndex);
56011
56012         if(pt == "after"){
56013             newIndex++;
56014         }
56015         if(oldIndex < newIndex){
56016             newIndex--;
56017         }
56018         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
56019             return false;
56020         }
56021         px +=  this.proxyOffsets[0];
56022         this.proxyTop.setLeftTop(px, py);
56023         this.proxyTop.show();
56024         if(!this.bottomOffset){
56025             this.bottomOffset = this.view.mainHd.getHeight();
56026         }
56027         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
56028         this.proxyBottom.show();
56029         return pt;
56030     },
56031
56032     onNodeEnter : function(n, dd, e, data){
56033         if(data.header != n){
56034             this.positionIndicator(data.header, n, e);
56035         }
56036     },
56037
56038     onNodeOver : function(n, dd, e, data){
56039         var result = false;
56040         if(data.header != n){
56041             result = this.positionIndicator(data.header, n, e);
56042         }
56043         if(!result){
56044             this.proxyTop.hide();
56045             this.proxyBottom.hide();
56046         }
56047         return result ? this.dropAllowed : this.dropNotAllowed;
56048     },
56049
56050     onNodeOut : function(n, dd, e, data){
56051         this.proxyTop.hide();
56052         this.proxyBottom.hide();
56053     },
56054
56055     onNodeDrop : function(n, dd, e, data){
56056         var h = data.header;
56057         if(h != n){
56058             var cm = this.grid.colModel;
56059             var x = Roo.lib.Event.getPageX(e);
56060             var r = Roo.lib.Dom.getRegion(n.firstChild);
56061             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
56062             var oldIndex = this.view.getCellIndex(h);
56063             var newIndex = this.view.getCellIndex(n);
56064             var locked = cm.isLocked(newIndex);
56065             if(pt == "after"){
56066                 newIndex++;
56067             }
56068             if(oldIndex < newIndex){
56069                 newIndex--;
56070             }
56071             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
56072                 return false;
56073             }
56074             cm.setLocked(oldIndex, locked, true);
56075             cm.moveColumn(oldIndex, newIndex);
56076             this.grid.fireEvent("columnmove", oldIndex, newIndex);
56077             return true;
56078         }
56079         return false;
56080     }
56081 });
56082 /*
56083  * Based on:
56084  * Ext JS Library 1.1.1
56085  * Copyright(c) 2006-2007, Ext JS, LLC.
56086  *
56087  * Originally Released Under LGPL - original licence link has changed is not relivant.
56088  *
56089  * Fork - LGPL
56090  * <script type="text/javascript">
56091  */
56092   
56093 /**
56094  * @class Roo.grid.GridView
56095  * @extends Roo.util.Observable
56096  *
56097  * @constructor
56098  * @param {Object} config
56099  */
56100 Roo.grid.GridView = function(config){
56101     Roo.grid.GridView.superclass.constructor.call(this);
56102     this.el = null;
56103
56104     Roo.apply(this, config);
56105 };
56106
56107 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
56108
56109     unselectable :  'unselectable="on"',
56110     unselectableCls :  'x-unselectable',
56111     
56112     
56113     rowClass : "x-grid-row",
56114
56115     cellClass : "x-grid-col",
56116
56117     tdClass : "x-grid-td",
56118
56119     hdClass : "x-grid-hd",
56120
56121     splitClass : "x-grid-split",
56122
56123     sortClasses : ["sort-asc", "sort-desc"],
56124
56125     enableMoveAnim : false,
56126
56127     hlColor: "C3DAF9",
56128
56129     dh : Roo.DomHelper,
56130
56131     fly : Roo.Element.fly,
56132
56133     css : Roo.util.CSS,
56134
56135     borderWidth: 1,
56136
56137     splitOffset: 3,
56138
56139     scrollIncrement : 22,
56140
56141     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
56142
56143     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
56144
56145     bind : function(ds, cm){
56146         if(this.ds){
56147             this.ds.un("load", this.onLoad, this);
56148             this.ds.un("datachanged", this.onDataChange, this);
56149             this.ds.un("add", this.onAdd, this);
56150             this.ds.un("remove", this.onRemove, this);
56151             this.ds.un("update", this.onUpdate, this);
56152             this.ds.un("clear", this.onClear, this);
56153         }
56154         if(ds){
56155             ds.on("load", this.onLoad, this);
56156             ds.on("datachanged", this.onDataChange, this);
56157             ds.on("add", this.onAdd, this);
56158             ds.on("remove", this.onRemove, this);
56159             ds.on("update", this.onUpdate, this);
56160             ds.on("clear", this.onClear, this);
56161         }
56162         this.ds = ds;
56163
56164         if(this.cm){
56165             this.cm.un("widthchange", this.onColWidthChange, this);
56166             this.cm.un("headerchange", this.onHeaderChange, this);
56167             this.cm.un("hiddenchange", this.onHiddenChange, this);
56168             this.cm.un("columnmoved", this.onColumnMove, this);
56169             this.cm.un("columnlockchange", this.onColumnLock, this);
56170         }
56171         if(cm){
56172             this.generateRules(cm);
56173             cm.on("widthchange", this.onColWidthChange, this);
56174             cm.on("headerchange", this.onHeaderChange, this);
56175             cm.on("hiddenchange", this.onHiddenChange, this);
56176             cm.on("columnmoved", this.onColumnMove, this);
56177             cm.on("columnlockchange", this.onColumnLock, this);
56178         }
56179         this.cm = cm;
56180     },
56181
56182     init: function(grid){
56183         Roo.grid.GridView.superclass.init.call(this, grid);
56184
56185         this.bind(grid.dataSource, grid.colModel);
56186
56187         grid.on("headerclick", this.handleHeaderClick, this);
56188
56189         if(grid.trackMouseOver){
56190             grid.on("mouseover", this.onRowOver, this);
56191             grid.on("mouseout", this.onRowOut, this);
56192         }
56193         grid.cancelTextSelection = function(){};
56194         this.gridId = grid.id;
56195
56196         var tpls = this.templates || {};
56197
56198         if(!tpls.master){
56199             tpls.master = new Roo.Template(
56200                '<div class="x-grid" hidefocus="true">',
56201                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56202                   '<div class="x-grid-topbar"></div>',
56203                   '<div class="x-grid-scroller"><div></div></div>',
56204                   '<div class="x-grid-locked">',
56205                       '<div class="x-grid-header">{lockedHeader}</div>',
56206                       '<div class="x-grid-body">{lockedBody}</div>',
56207                   "</div>",
56208                   '<div class="x-grid-viewport">',
56209                       '<div class="x-grid-header">{header}</div>',
56210                       '<div class="x-grid-body">{body}</div>',
56211                   "</div>",
56212                   '<div class="x-grid-bottombar"></div>',
56213                  
56214                   '<div class="x-grid-resize-proxy">&#160;</div>',
56215                "</div>"
56216             );
56217             tpls.master.disableformats = true;
56218         }
56219
56220         if(!tpls.header){
56221             tpls.header = new Roo.Template(
56222                '<table border="0" cellspacing="0" cellpadding="0">',
56223                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56224                "</table>{splits}"
56225             );
56226             tpls.header.disableformats = true;
56227         }
56228         tpls.header.compile();
56229
56230         if(!tpls.hcell){
56231             tpls.hcell = new Roo.Template(
56232                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56233                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56234                 "</div></td>"
56235              );
56236              tpls.hcell.disableFormats = true;
56237         }
56238         tpls.hcell.compile();
56239
56240         if(!tpls.hsplit){
56241             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56242                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56243             tpls.hsplit.disableFormats = true;
56244         }
56245         tpls.hsplit.compile();
56246
56247         if(!tpls.body){
56248             tpls.body = new Roo.Template(
56249                '<table border="0" cellspacing="0" cellpadding="0">',
56250                "<tbody>{rows}</tbody>",
56251                "</table>"
56252             );
56253             tpls.body.disableFormats = true;
56254         }
56255         tpls.body.compile();
56256
56257         if(!tpls.row){
56258             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56259             tpls.row.disableFormats = true;
56260         }
56261         tpls.row.compile();
56262
56263         if(!tpls.cell){
56264             tpls.cell = new Roo.Template(
56265                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56266                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56267                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56268                 "</td>"
56269             );
56270             tpls.cell.disableFormats = true;
56271         }
56272         tpls.cell.compile();
56273
56274         this.templates = tpls;
56275     },
56276
56277     // remap these for backwards compat
56278     onColWidthChange : function(){
56279         this.updateColumns.apply(this, arguments);
56280     },
56281     onHeaderChange : function(){
56282         this.updateHeaders.apply(this, arguments);
56283     }, 
56284     onHiddenChange : function(){
56285         this.handleHiddenChange.apply(this, arguments);
56286     },
56287     onColumnMove : function(){
56288         this.handleColumnMove.apply(this, arguments);
56289     },
56290     onColumnLock : function(){
56291         this.handleLockChange.apply(this, arguments);
56292     },
56293
56294     onDataChange : function(){
56295         this.refresh();
56296         this.updateHeaderSortState();
56297     },
56298
56299     onClear : function(){
56300         this.refresh();
56301     },
56302
56303     onUpdate : function(ds, record){
56304         this.refreshRow(record);
56305     },
56306
56307     refreshRow : function(record){
56308         var ds = this.ds, index;
56309         if(typeof record == 'number'){
56310             index = record;
56311             record = ds.getAt(index);
56312         }else{
56313             index = ds.indexOf(record);
56314         }
56315         this.insertRows(ds, index, index, true);
56316         this.onRemove(ds, record, index+1, true);
56317         this.syncRowHeights(index, index);
56318         this.layout();
56319         this.fireEvent("rowupdated", this, index, record);
56320     },
56321
56322     onAdd : function(ds, records, index){
56323         this.insertRows(ds, index, index + (records.length-1));
56324     },
56325
56326     onRemove : function(ds, record, index, isUpdate){
56327         if(isUpdate !== true){
56328             this.fireEvent("beforerowremoved", this, index, record);
56329         }
56330         var bt = this.getBodyTable(), lt = this.getLockedTable();
56331         if(bt.rows[index]){
56332             bt.firstChild.removeChild(bt.rows[index]);
56333         }
56334         if(lt.rows[index]){
56335             lt.firstChild.removeChild(lt.rows[index]);
56336         }
56337         if(isUpdate !== true){
56338             this.stripeRows(index);
56339             this.syncRowHeights(index, index);
56340             this.layout();
56341             this.fireEvent("rowremoved", this, index, record);
56342         }
56343     },
56344
56345     onLoad : function(){
56346         this.scrollToTop();
56347     },
56348
56349     /**
56350      * Scrolls the grid to the top
56351      */
56352     scrollToTop : function(){
56353         if(this.scroller){
56354             this.scroller.dom.scrollTop = 0;
56355             this.syncScroll();
56356         }
56357     },
56358
56359     /**
56360      * Gets a panel in the header of the grid that can be used for toolbars etc.
56361      * After modifying the contents of this panel a call to grid.autoSize() may be
56362      * required to register any changes in size.
56363      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56364      * @return Roo.Element
56365      */
56366     getHeaderPanel : function(doShow){
56367         if(doShow){
56368             this.headerPanel.show();
56369         }
56370         return this.headerPanel;
56371     },
56372
56373     /**
56374      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56375      * After modifying the contents of this panel a call to grid.autoSize() may be
56376      * required to register any changes in size.
56377      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56378      * @return Roo.Element
56379      */
56380     getFooterPanel : function(doShow){
56381         if(doShow){
56382             this.footerPanel.show();
56383         }
56384         return this.footerPanel;
56385     },
56386
56387     initElements : function(){
56388         var E = Roo.Element;
56389         var el = this.grid.getGridEl().dom.firstChild;
56390         var cs = el.childNodes;
56391
56392         this.el = new E(el);
56393         
56394          this.focusEl = new E(el.firstChild);
56395         this.focusEl.swallowEvent("click", true);
56396         
56397         this.headerPanel = new E(cs[1]);
56398         this.headerPanel.enableDisplayMode("block");
56399
56400         this.scroller = new E(cs[2]);
56401         this.scrollSizer = new E(this.scroller.dom.firstChild);
56402
56403         this.lockedWrap = new E(cs[3]);
56404         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56405         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56406
56407         this.mainWrap = new E(cs[4]);
56408         this.mainHd = new E(this.mainWrap.dom.firstChild);
56409         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56410
56411         this.footerPanel = new E(cs[5]);
56412         this.footerPanel.enableDisplayMode("block");
56413
56414         this.resizeProxy = new E(cs[6]);
56415
56416         this.headerSelector = String.format(
56417            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56418            this.lockedHd.id, this.mainHd.id
56419         );
56420
56421         this.splitterSelector = String.format(
56422            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56423            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56424         );
56425     },
56426     idToCssName : function(s)
56427     {
56428         return s.replace(/[^a-z0-9]+/ig, '-');
56429     },
56430
56431     getHeaderCell : function(index){
56432         return Roo.DomQuery.select(this.headerSelector)[index];
56433     },
56434
56435     getHeaderCellMeasure : function(index){
56436         return this.getHeaderCell(index).firstChild;
56437     },
56438
56439     getHeaderCellText : function(index){
56440         return this.getHeaderCell(index).firstChild.firstChild;
56441     },
56442
56443     getLockedTable : function(){
56444         return this.lockedBody.dom.firstChild;
56445     },
56446
56447     getBodyTable : function(){
56448         return this.mainBody.dom.firstChild;
56449     },
56450
56451     getLockedRow : function(index){
56452         return this.getLockedTable().rows[index];
56453     },
56454
56455     getRow : function(index){
56456         return this.getBodyTable().rows[index];
56457     },
56458
56459     getRowComposite : function(index){
56460         if(!this.rowEl){
56461             this.rowEl = new Roo.CompositeElementLite();
56462         }
56463         var els = [], lrow, mrow;
56464         if(lrow = this.getLockedRow(index)){
56465             els.push(lrow);
56466         }
56467         if(mrow = this.getRow(index)){
56468             els.push(mrow);
56469         }
56470         this.rowEl.elements = els;
56471         return this.rowEl;
56472     },
56473     /**
56474      * Gets the 'td' of the cell
56475      * 
56476      * @param {Integer} rowIndex row to select
56477      * @param {Integer} colIndex column to select
56478      * 
56479      * @return {Object} 
56480      */
56481     getCell : function(rowIndex, colIndex){
56482         var locked = this.cm.getLockedCount();
56483         var source;
56484         if(colIndex < locked){
56485             source = this.lockedBody.dom.firstChild;
56486         }else{
56487             source = this.mainBody.dom.firstChild;
56488             colIndex -= locked;
56489         }
56490         return source.rows[rowIndex].childNodes[colIndex];
56491     },
56492
56493     getCellText : function(rowIndex, colIndex){
56494         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56495     },
56496
56497     getCellBox : function(cell){
56498         var b = this.fly(cell).getBox();
56499         if(Roo.isOpera){ // opera fails to report the Y
56500             b.y = cell.offsetTop + this.mainBody.getY();
56501         }
56502         return b;
56503     },
56504
56505     getCellIndex : function(cell){
56506         var id = String(cell.className).match(this.cellRE);
56507         if(id){
56508             return parseInt(id[1], 10);
56509         }
56510         return 0;
56511     },
56512
56513     findHeaderIndex : function(n){
56514         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56515         return r ? this.getCellIndex(r) : false;
56516     },
56517
56518     findHeaderCell : function(n){
56519         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56520         return r ? r : false;
56521     },
56522
56523     findRowIndex : function(n){
56524         if(!n){
56525             return false;
56526         }
56527         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56528         return r ? r.rowIndex : false;
56529     },
56530
56531     findCellIndex : function(node){
56532         var stop = this.el.dom;
56533         while(node && node != stop){
56534             if(this.findRE.test(node.className)){
56535                 return this.getCellIndex(node);
56536             }
56537             node = node.parentNode;
56538         }
56539         return false;
56540     },
56541
56542     getColumnId : function(index){
56543         return this.cm.getColumnId(index);
56544     },
56545
56546     getSplitters : function()
56547     {
56548         if(this.splitterSelector){
56549            return Roo.DomQuery.select(this.splitterSelector);
56550         }else{
56551             return null;
56552       }
56553     },
56554
56555     getSplitter : function(index){
56556         return this.getSplitters()[index];
56557     },
56558
56559     onRowOver : function(e, t){
56560         var row;
56561         if((row = this.findRowIndex(t)) !== false){
56562             this.getRowComposite(row).addClass("x-grid-row-over");
56563         }
56564     },
56565
56566     onRowOut : function(e, t){
56567         var row;
56568         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56569             this.getRowComposite(row).removeClass("x-grid-row-over");
56570         }
56571     },
56572
56573     renderHeaders : function(){
56574         var cm = this.cm;
56575         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56576         var cb = [], lb = [], sb = [], lsb = [], p = {};
56577         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56578             p.cellId = "x-grid-hd-0-" + i;
56579             p.splitId = "x-grid-csplit-0-" + i;
56580             p.id = cm.getColumnId(i);
56581             p.value = cm.getColumnHeader(i) || "";
56582             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56583             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56584             if(!cm.isLocked(i)){
56585                 cb[cb.length] = ct.apply(p);
56586                 sb[sb.length] = st.apply(p);
56587             }else{
56588                 lb[lb.length] = ct.apply(p);
56589                 lsb[lsb.length] = st.apply(p);
56590             }
56591         }
56592         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56593                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56594     },
56595
56596     updateHeaders : function(){
56597         var html = this.renderHeaders();
56598         this.lockedHd.update(html[0]);
56599         this.mainHd.update(html[1]);
56600     },
56601
56602     /**
56603      * Focuses the specified row.
56604      * @param {Number} row The row index
56605      */
56606     focusRow : function(row)
56607     {
56608         //Roo.log('GridView.focusRow');
56609         var x = this.scroller.dom.scrollLeft;
56610         this.focusCell(row, 0, false);
56611         this.scroller.dom.scrollLeft = x;
56612     },
56613
56614     /**
56615      * Focuses the specified cell.
56616      * @param {Number} row The row index
56617      * @param {Number} col The column index
56618      * @param {Boolean} hscroll false to disable horizontal scrolling
56619      */
56620     focusCell : function(row, col, hscroll)
56621     {
56622         //Roo.log('GridView.focusCell');
56623         var el = this.ensureVisible(row, col, hscroll);
56624         this.focusEl.alignTo(el, "tl-tl");
56625         if(Roo.isGecko){
56626             this.focusEl.focus();
56627         }else{
56628             this.focusEl.focus.defer(1, this.focusEl);
56629         }
56630     },
56631
56632     /**
56633      * Scrolls the specified cell into view
56634      * @param {Number} row The row index
56635      * @param {Number} col The column index
56636      * @param {Boolean} hscroll false to disable horizontal scrolling
56637      */
56638     ensureVisible : function(row, col, hscroll)
56639     {
56640         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56641         //return null; //disable for testing.
56642         if(typeof row != "number"){
56643             row = row.rowIndex;
56644         }
56645         if(row < 0 && row >= this.ds.getCount()){
56646             return  null;
56647         }
56648         col = (col !== undefined ? col : 0);
56649         var cm = this.grid.colModel;
56650         while(cm.isHidden(col)){
56651             col++;
56652         }
56653
56654         var el = this.getCell(row, col);
56655         if(!el){
56656             return null;
56657         }
56658         var c = this.scroller.dom;
56659
56660         var ctop = parseInt(el.offsetTop, 10);
56661         var cleft = parseInt(el.offsetLeft, 10);
56662         var cbot = ctop + el.offsetHeight;
56663         var cright = cleft + el.offsetWidth;
56664         
56665         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56666         var stop = parseInt(c.scrollTop, 10);
56667         var sleft = parseInt(c.scrollLeft, 10);
56668         var sbot = stop + ch;
56669         var sright = sleft + c.clientWidth;
56670         /*
56671         Roo.log('GridView.ensureVisible:' +
56672                 ' ctop:' + ctop +
56673                 ' c.clientHeight:' + c.clientHeight +
56674                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56675                 ' stop:' + stop +
56676                 ' cbot:' + cbot +
56677                 ' sbot:' + sbot +
56678                 ' ch:' + ch  
56679                 );
56680         */
56681         if(ctop < stop){
56682              c.scrollTop = ctop;
56683             //Roo.log("set scrolltop to ctop DISABLE?");
56684         }else if(cbot > sbot){
56685             //Roo.log("set scrolltop to cbot-ch");
56686             c.scrollTop = cbot-ch;
56687         }
56688         
56689         if(hscroll !== false){
56690             if(cleft < sleft){
56691                 c.scrollLeft = cleft;
56692             }else if(cright > sright){
56693                 c.scrollLeft = cright-c.clientWidth;
56694             }
56695         }
56696          
56697         return el;
56698     },
56699
56700     updateColumns : function(){
56701         this.grid.stopEditing();
56702         var cm = this.grid.colModel, colIds = this.getColumnIds();
56703         //var totalWidth = cm.getTotalWidth();
56704         var pos = 0;
56705         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56706             //if(cm.isHidden(i)) continue;
56707             var w = cm.getColumnWidth(i);
56708             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56709             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56710         }
56711         this.updateSplitters();
56712     },
56713
56714     generateRules : function(cm){
56715         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56716         Roo.util.CSS.removeStyleSheet(rulesId);
56717         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56718             var cid = cm.getColumnId(i);
56719             var align = '';
56720             if(cm.config[i].align){
56721                 align = 'text-align:'+cm.config[i].align+';';
56722             }
56723             var hidden = '';
56724             if(cm.isHidden(i)){
56725                 hidden = 'display:none;';
56726             }
56727             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56728             ruleBuf.push(
56729                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56730                     this.hdSelector, cid, " {\n", align, width, "}\n",
56731                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56732                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56733         }
56734         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56735     },
56736
56737     updateSplitters : function(){
56738         var cm = this.cm, s = this.getSplitters();
56739         if(s){ // splitters not created yet
56740             var pos = 0, locked = true;
56741             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56742                 if(cm.isHidden(i)) {
56743                     continue;
56744                 }
56745                 var w = cm.getColumnWidth(i); // make sure it's a number
56746                 if(!cm.isLocked(i) && locked){
56747                     pos = 0;
56748                     locked = false;
56749                 }
56750                 pos += w;
56751                 s[i].style.left = (pos-this.splitOffset) + "px";
56752             }
56753         }
56754     },
56755
56756     handleHiddenChange : function(colModel, colIndex, hidden){
56757         if(hidden){
56758             this.hideColumn(colIndex);
56759         }else{
56760             this.unhideColumn(colIndex);
56761         }
56762     },
56763
56764     hideColumn : function(colIndex){
56765         var cid = this.getColumnId(colIndex);
56766         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56767         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56768         if(Roo.isSafari){
56769             this.updateHeaders();
56770         }
56771         this.updateSplitters();
56772         this.layout();
56773     },
56774
56775     unhideColumn : function(colIndex){
56776         var cid = this.getColumnId(colIndex);
56777         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56778         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56779
56780         if(Roo.isSafari){
56781             this.updateHeaders();
56782         }
56783         this.updateSplitters();
56784         this.layout();
56785     },
56786
56787     insertRows : function(dm, firstRow, lastRow, isUpdate){
56788         if(firstRow == 0 && lastRow == dm.getCount()-1){
56789             this.refresh();
56790         }else{
56791             if(!isUpdate){
56792                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56793             }
56794             var s = this.getScrollState();
56795             var markup = this.renderRows(firstRow, lastRow);
56796             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56797             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56798             this.restoreScroll(s);
56799             if(!isUpdate){
56800                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56801                 this.syncRowHeights(firstRow, lastRow);
56802                 this.stripeRows(firstRow);
56803                 this.layout();
56804             }
56805         }
56806     },
56807
56808     bufferRows : function(markup, target, index){
56809         var before = null, trows = target.rows, tbody = target.tBodies[0];
56810         if(index < trows.length){
56811             before = trows[index];
56812         }
56813         var b = document.createElement("div");
56814         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56815         var rows = b.firstChild.rows;
56816         for(var i = 0, len = rows.length; i < len; i++){
56817             if(before){
56818                 tbody.insertBefore(rows[0], before);
56819             }else{
56820                 tbody.appendChild(rows[0]);
56821             }
56822         }
56823         b.innerHTML = "";
56824         b = null;
56825     },
56826
56827     deleteRows : function(dm, firstRow, lastRow){
56828         if(dm.getRowCount()<1){
56829             this.fireEvent("beforerefresh", this);
56830             this.mainBody.update("");
56831             this.lockedBody.update("");
56832             this.fireEvent("refresh", this);
56833         }else{
56834             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56835             var bt = this.getBodyTable();
56836             var tbody = bt.firstChild;
56837             var rows = bt.rows;
56838             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56839                 tbody.removeChild(rows[firstRow]);
56840             }
56841             this.stripeRows(firstRow);
56842             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56843         }
56844     },
56845
56846     updateRows : function(dataSource, firstRow, lastRow){
56847         var s = this.getScrollState();
56848         this.refresh();
56849         this.restoreScroll(s);
56850     },
56851
56852     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56853         if(!noRefresh){
56854            this.refresh();
56855         }
56856         this.updateHeaderSortState();
56857     },
56858
56859     getScrollState : function(){
56860         
56861         var sb = this.scroller.dom;
56862         return {left: sb.scrollLeft, top: sb.scrollTop};
56863     },
56864
56865     stripeRows : function(startRow){
56866         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56867             return;
56868         }
56869         startRow = startRow || 0;
56870         var rows = this.getBodyTable().rows;
56871         var lrows = this.getLockedTable().rows;
56872         var cls = ' x-grid-row-alt ';
56873         for(var i = startRow, len = rows.length; i < len; i++){
56874             var row = rows[i], lrow = lrows[i];
56875             var isAlt = ((i+1) % 2 == 0);
56876             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56877             if(isAlt == hasAlt){
56878                 continue;
56879             }
56880             if(isAlt){
56881                 row.className += " x-grid-row-alt";
56882             }else{
56883                 row.className = row.className.replace("x-grid-row-alt", "");
56884             }
56885             if(lrow){
56886                 lrow.className = row.className;
56887             }
56888         }
56889     },
56890
56891     restoreScroll : function(state){
56892         //Roo.log('GridView.restoreScroll');
56893         var sb = this.scroller.dom;
56894         sb.scrollLeft = state.left;
56895         sb.scrollTop = state.top;
56896         this.syncScroll();
56897     },
56898
56899     syncScroll : function(){
56900         //Roo.log('GridView.syncScroll');
56901         var sb = this.scroller.dom;
56902         var sh = this.mainHd.dom;
56903         var bs = this.mainBody.dom;
56904         var lv = this.lockedBody.dom;
56905         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56906         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56907     },
56908
56909     handleScroll : function(e){
56910         this.syncScroll();
56911         var sb = this.scroller.dom;
56912         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56913         e.stopEvent();
56914     },
56915
56916     handleWheel : function(e){
56917         var d = e.getWheelDelta();
56918         this.scroller.dom.scrollTop -= d*22;
56919         // set this here to prevent jumpy scrolling on large tables
56920         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56921         e.stopEvent();
56922     },
56923
56924     renderRows : function(startRow, endRow){
56925         // pull in all the crap needed to render rows
56926         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56927         var colCount = cm.getColumnCount();
56928
56929         if(ds.getCount() < 1){
56930             return ["", ""];
56931         }
56932
56933         // build a map for all the columns
56934         var cs = [];
56935         for(var i = 0; i < colCount; i++){
56936             var name = cm.getDataIndex(i);
56937             cs[i] = {
56938                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56939                 renderer : cm.getRenderer(i),
56940                 id : cm.getColumnId(i),
56941                 locked : cm.isLocked(i),
56942                 has_editor : cm.isCellEditable(i)
56943             };
56944         }
56945
56946         startRow = startRow || 0;
56947         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56948
56949         // records to render
56950         var rs = ds.getRange(startRow, endRow);
56951
56952         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56953     },
56954
56955     // As much as I hate to duplicate code, this was branched because FireFox really hates
56956     // [].join("") on strings. The performance difference was substantial enough to
56957     // branch this function
56958     doRender : Roo.isGecko ?
56959             function(cs, rs, ds, startRow, colCount, stripe){
56960                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56961                 // buffers
56962                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56963                 
56964                 var hasListener = this.grid.hasListener('rowclass');
56965                 var rowcfg = {};
56966                 for(var j = 0, len = rs.length; j < len; j++){
56967                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56968                     for(var i = 0; i < colCount; i++){
56969                         c = cs[i];
56970                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56971                         p.id = c.id;
56972                         p.css = p.attr = "";
56973                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56974                         if(p.value == undefined || p.value === "") {
56975                             p.value = "&#160;";
56976                         }
56977                         if(c.has_editor){
56978                             p.css += ' x-grid-editable-cell';
56979                         }
56980                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56981                             p.css +=  ' x-grid-dirty-cell';
56982                         }
56983                         var markup = ct.apply(p);
56984                         if(!c.locked){
56985                             cb+= markup;
56986                         }else{
56987                             lcb+= markup;
56988                         }
56989                     }
56990                     var alt = [];
56991                     if(stripe && ((rowIndex+1) % 2 == 0)){
56992                         alt.push("x-grid-row-alt")
56993                     }
56994                     if(r.dirty){
56995                         alt.push(  " x-grid-dirty-row");
56996                     }
56997                     rp.cells = lcb;
56998                     if(this.getRowClass){
56999                         alt.push(this.getRowClass(r, rowIndex));
57000                     }
57001                     if (hasListener) {
57002                         rowcfg = {
57003                              
57004                             record: r,
57005                             rowIndex : rowIndex,
57006                             rowClass : ''
57007                         };
57008                         this.grid.fireEvent('rowclass', this, rowcfg);
57009                         alt.push(rowcfg.rowClass);
57010                     }
57011                     rp.alt = alt.join(" ");
57012                     lbuf+= rt.apply(rp);
57013                     rp.cells = cb;
57014                     buf+=  rt.apply(rp);
57015                 }
57016                 return [lbuf, buf];
57017             } :
57018             function(cs, rs, ds, startRow, colCount, stripe){
57019                 var ts = this.templates, ct = ts.cell, rt = ts.row;
57020                 // buffers
57021                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
57022                 var hasListener = this.grid.hasListener('rowclass');
57023  
57024                 var rowcfg = {};
57025                 for(var j = 0, len = rs.length; j < len; j++){
57026                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
57027                     for(var i = 0; i < colCount; i++){
57028                         c = cs[i];
57029                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
57030                         p.id = c.id;
57031                         p.css = p.attr = "";
57032                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
57033                         if(p.value == undefined || p.value === "") {
57034                             p.value = "&#160;";
57035                         }
57036                         //Roo.log(c);
57037                          if(c.has_editor){
57038                             p.css += ' x-grid-editable-cell';
57039                         }
57040                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
57041                             p.css += ' x-grid-dirty-cell' 
57042                         }
57043                         
57044                         var markup = ct.apply(p);
57045                         if(!c.locked){
57046                             cb[cb.length] = markup;
57047                         }else{
57048                             lcb[lcb.length] = markup;
57049                         }
57050                     }
57051                     var alt = [];
57052                     if(stripe && ((rowIndex+1) % 2 == 0)){
57053                         alt.push( "x-grid-row-alt");
57054                     }
57055                     if(r.dirty){
57056                         alt.push(" x-grid-dirty-row");
57057                     }
57058                     rp.cells = lcb;
57059                     if(this.getRowClass){
57060                         alt.push( this.getRowClass(r, rowIndex));
57061                     }
57062                     if (hasListener) {
57063                         rowcfg = {
57064                              
57065                             record: r,
57066                             rowIndex : rowIndex,
57067                             rowClass : ''
57068                         };
57069                         this.grid.fireEvent('rowclass', this, rowcfg);
57070                         alt.push(rowcfg.rowClass);
57071                     }
57072                     
57073                     rp.alt = alt.join(" ");
57074                     rp.cells = lcb.join("");
57075                     lbuf[lbuf.length] = rt.apply(rp);
57076                     rp.cells = cb.join("");
57077                     buf[buf.length] =  rt.apply(rp);
57078                 }
57079                 return [lbuf.join(""), buf.join("")];
57080             },
57081
57082     renderBody : function(){
57083         var markup = this.renderRows();
57084         var bt = this.templates.body;
57085         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
57086     },
57087
57088     /**
57089      * Refreshes the grid
57090      * @param {Boolean} headersToo
57091      */
57092     refresh : function(headersToo){
57093         this.fireEvent("beforerefresh", this);
57094         this.grid.stopEditing();
57095         var result = this.renderBody();
57096         this.lockedBody.update(result[0]);
57097         this.mainBody.update(result[1]);
57098         if(headersToo === true){
57099             this.updateHeaders();
57100             this.updateColumns();
57101             this.updateSplitters();
57102             this.updateHeaderSortState();
57103         }
57104         this.syncRowHeights();
57105         this.layout();
57106         this.fireEvent("refresh", this);
57107     },
57108
57109     handleColumnMove : function(cm, oldIndex, newIndex){
57110         this.indexMap = null;
57111         var s = this.getScrollState();
57112         this.refresh(true);
57113         this.restoreScroll(s);
57114         this.afterMove(newIndex);
57115     },
57116
57117     afterMove : function(colIndex){
57118         if(this.enableMoveAnim && Roo.enableFx){
57119             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
57120         }
57121         // if multisort - fix sortOrder, and reload..
57122         if (this.grid.dataSource.multiSort) {
57123             // the we can call sort again..
57124             var dm = this.grid.dataSource;
57125             var cm = this.grid.colModel;
57126             var so = [];
57127             for(var i = 0; i < cm.config.length; i++ ) {
57128                 
57129                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
57130                     continue; // dont' bother, it's not in sort list or being set.
57131                 }
57132                 
57133                 so.push(cm.config[i].dataIndex);
57134             };
57135             dm.sortOrder = so;
57136             dm.load(dm.lastOptions);
57137             
57138             
57139         }
57140         
57141     },
57142
57143     updateCell : function(dm, rowIndex, dataIndex){
57144         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
57145         if(typeof colIndex == "undefined"){ // not present in grid
57146             return;
57147         }
57148         var cm = this.grid.colModel;
57149         var cell = this.getCell(rowIndex, colIndex);
57150         var cellText = this.getCellText(rowIndex, colIndex);
57151
57152         var p = {
57153             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
57154             id : cm.getColumnId(colIndex),
57155             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
57156         };
57157         var renderer = cm.getRenderer(colIndex);
57158         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
57159         if(typeof val == "undefined" || val === "") {
57160             val = "&#160;";
57161         }
57162         cellText.innerHTML = val;
57163         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
57164         this.syncRowHeights(rowIndex, rowIndex);
57165     },
57166
57167     calcColumnWidth : function(colIndex, maxRowsToMeasure){
57168         var maxWidth = 0;
57169         if(this.grid.autoSizeHeaders){
57170             var h = this.getHeaderCellMeasure(colIndex);
57171             maxWidth = Math.max(maxWidth, h.scrollWidth);
57172         }
57173         var tb, index;
57174         if(this.cm.isLocked(colIndex)){
57175             tb = this.getLockedTable();
57176             index = colIndex;
57177         }else{
57178             tb = this.getBodyTable();
57179             index = colIndex - this.cm.getLockedCount();
57180         }
57181         if(tb && tb.rows){
57182             var rows = tb.rows;
57183             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
57184             for(var i = 0; i < stopIndex; i++){
57185                 var cell = rows[i].childNodes[index].firstChild;
57186                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
57187             }
57188         }
57189         return maxWidth + /*margin for error in IE*/ 5;
57190     },
57191     /**
57192      * Autofit a column to its content.
57193      * @param {Number} colIndex
57194      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
57195      */
57196      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
57197          if(this.cm.isHidden(colIndex)){
57198              return; // can't calc a hidden column
57199          }
57200         if(forceMinSize){
57201             var cid = this.cm.getColumnId(colIndex);
57202             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57203            if(this.grid.autoSizeHeaders){
57204                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57205            }
57206         }
57207         var newWidth = this.calcColumnWidth(colIndex);
57208         this.cm.setColumnWidth(colIndex,
57209             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57210         if(!suppressEvent){
57211             this.grid.fireEvent("columnresize", colIndex, newWidth);
57212         }
57213     },
57214
57215     /**
57216      * Autofits all columns to their content and then expands to fit any extra space in the grid
57217      */
57218      autoSizeColumns : function(){
57219         var cm = this.grid.colModel;
57220         var colCount = cm.getColumnCount();
57221         for(var i = 0; i < colCount; i++){
57222             this.autoSizeColumn(i, true, true);
57223         }
57224         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57225             this.fitColumns();
57226         }else{
57227             this.updateColumns();
57228             this.layout();
57229         }
57230     },
57231
57232     /**
57233      * Autofits all columns to the grid's width proportionate with their current size
57234      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57235      */
57236     fitColumns : function(reserveScrollSpace){
57237         var cm = this.grid.colModel;
57238         var colCount = cm.getColumnCount();
57239         var cols = [];
57240         var width = 0;
57241         var i, w;
57242         for (i = 0; i < colCount; i++){
57243             if(!cm.isHidden(i) && !cm.isFixed(i)){
57244                 w = cm.getColumnWidth(i);
57245                 cols.push(i);
57246                 cols.push(w);
57247                 width += w;
57248             }
57249         }
57250         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57251         if(reserveScrollSpace){
57252             avail -= 17;
57253         }
57254         var frac = (avail - cm.getTotalWidth())/width;
57255         while (cols.length){
57256             w = cols.pop();
57257             i = cols.pop();
57258             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57259         }
57260         this.updateColumns();
57261         this.layout();
57262     },
57263
57264     onRowSelect : function(rowIndex){
57265         var row = this.getRowComposite(rowIndex);
57266         row.addClass("x-grid-row-selected");
57267     },
57268
57269     onRowDeselect : function(rowIndex){
57270         var row = this.getRowComposite(rowIndex);
57271         row.removeClass("x-grid-row-selected");
57272     },
57273
57274     onCellSelect : function(row, col){
57275         var cell = this.getCell(row, col);
57276         if(cell){
57277             Roo.fly(cell).addClass("x-grid-cell-selected");
57278         }
57279     },
57280
57281     onCellDeselect : function(row, col){
57282         var cell = this.getCell(row, col);
57283         if(cell){
57284             Roo.fly(cell).removeClass("x-grid-cell-selected");
57285         }
57286     },
57287
57288     updateHeaderSortState : function(){
57289         
57290         // sort state can be single { field: xxx, direction : yyy}
57291         // or   { xxx=>ASC , yyy : DESC ..... }
57292         
57293         var mstate = {};
57294         if (!this.ds.multiSort) { 
57295             var state = this.ds.getSortState();
57296             if(!state){
57297                 return;
57298             }
57299             mstate[state.field] = state.direction;
57300             // FIXME... - this is not used here.. but might be elsewhere..
57301             this.sortState = state;
57302             
57303         } else {
57304             mstate = this.ds.sortToggle;
57305         }
57306         //remove existing sort classes..
57307         
57308         var sc = this.sortClasses;
57309         var hds = this.el.select(this.headerSelector).removeClass(sc);
57310         
57311         for(var f in mstate) {
57312         
57313             var sortColumn = this.cm.findColumnIndex(f);
57314             
57315             if(sortColumn != -1){
57316                 var sortDir = mstate[f];        
57317                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57318             }
57319         }
57320         
57321          
57322         
57323     },
57324
57325
57326     handleHeaderClick : function(g, index,e){
57327         
57328         Roo.log("header click");
57329         
57330         if (Roo.isTouch) {
57331             // touch events on header are handled by context
57332             this.handleHdCtx(g,index,e);
57333             return;
57334         }
57335         
57336         
57337         if(this.headersDisabled){
57338             return;
57339         }
57340         var dm = g.dataSource, cm = g.colModel;
57341         if(!cm.isSortable(index)){
57342             return;
57343         }
57344         g.stopEditing();
57345         
57346         if (dm.multiSort) {
57347             // update the sortOrder
57348             var so = [];
57349             for(var i = 0; i < cm.config.length; i++ ) {
57350                 
57351                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57352                     continue; // dont' bother, it's not in sort list or being set.
57353                 }
57354                 
57355                 so.push(cm.config[i].dataIndex);
57356             };
57357             dm.sortOrder = so;
57358         }
57359         
57360         
57361         dm.sort(cm.getDataIndex(index));
57362     },
57363
57364
57365     destroy : function(){
57366         if(this.colMenu){
57367             this.colMenu.removeAll();
57368             Roo.menu.MenuMgr.unregister(this.colMenu);
57369             this.colMenu.getEl().remove();
57370             delete this.colMenu;
57371         }
57372         if(this.hmenu){
57373             this.hmenu.removeAll();
57374             Roo.menu.MenuMgr.unregister(this.hmenu);
57375             this.hmenu.getEl().remove();
57376             delete this.hmenu;
57377         }
57378         if(this.grid.enableColumnMove){
57379             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57380             if(dds){
57381                 for(var dd in dds){
57382                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57383                         var elid = dds[dd].dragElId;
57384                         dds[dd].unreg();
57385                         Roo.get(elid).remove();
57386                     } else if(dds[dd].config.isTarget){
57387                         dds[dd].proxyTop.remove();
57388                         dds[dd].proxyBottom.remove();
57389                         dds[dd].unreg();
57390                     }
57391                     if(Roo.dd.DDM.locationCache[dd]){
57392                         delete Roo.dd.DDM.locationCache[dd];
57393                     }
57394                 }
57395                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57396             }
57397         }
57398         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57399         this.bind(null, null);
57400         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57401     },
57402
57403     handleLockChange : function(){
57404         this.refresh(true);
57405     },
57406
57407     onDenyColumnLock : function(){
57408
57409     },
57410
57411     onDenyColumnHide : function(){
57412
57413     },
57414
57415     handleHdMenuClick : function(item){
57416         var index = this.hdCtxIndex;
57417         var cm = this.cm, ds = this.ds;
57418         switch(item.id){
57419             case "asc":
57420                 ds.sort(cm.getDataIndex(index), "ASC");
57421                 break;
57422             case "desc":
57423                 ds.sort(cm.getDataIndex(index), "DESC");
57424                 break;
57425             case "lock":
57426                 var lc = cm.getLockedCount();
57427                 if(cm.getColumnCount(true) <= lc+1){
57428                     this.onDenyColumnLock();
57429                     return;
57430                 }
57431                 if(lc != index){
57432                     cm.setLocked(index, true, true);
57433                     cm.moveColumn(index, lc);
57434                     this.grid.fireEvent("columnmove", index, lc);
57435                 }else{
57436                     cm.setLocked(index, true);
57437                 }
57438             break;
57439             case "unlock":
57440                 var lc = cm.getLockedCount();
57441                 if((lc-1) != index){
57442                     cm.setLocked(index, false, true);
57443                     cm.moveColumn(index, lc-1);
57444                     this.grid.fireEvent("columnmove", index, lc-1);
57445                 }else{
57446                     cm.setLocked(index, false);
57447                 }
57448             break;
57449             case 'wider': // used to expand cols on touch..
57450             case 'narrow':
57451                 var cw = cm.getColumnWidth(index);
57452                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57453                 cw = Math.max(0, cw);
57454                 cw = Math.min(cw,4000);
57455                 cm.setColumnWidth(index, cw);
57456                 break;
57457                 
57458             default:
57459                 index = cm.getIndexById(item.id.substr(4));
57460                 if(index != -1){
57461                     if(item.checked && cm.getColumnCount(true) <= 1){
57462                         this.onDenyColumnHide();
57463                         return false;
57464                     }
57465                     cm.setHidden(index, item.checked);
57466                 }
57467         }
57468         return true;
57469     },
57470
57471     beforeColMenuShow : function(){
57472         var cm = this.cm,  colCount = cm.getColumnCount();
57473         this.colMenu.removeAll();
57474         for(var i = 0; i < colCount; i++){
57475             this.colMenu.add(new Roo.menu.CheckItem({
57476                 id: "col-"+cm.getColumnId(i),
57477                 text: cm.getColumnHeader(i),
57478                 checked: !cm.isHidden(i),
57479                 hideOnClick:false
57480             }));
57481         }
57482     },
57483
57484     handleHdCtx : function(g, index, e){
57485         e.stopEvent();
57486         var hd = this.getHeaderCell(index);
57487         this.hdCtxIndex = index;
57488         var ms = this.hmenu.items, cm = this.cm;
57489         ms.get("asc").setDisabled(!cm.isSortable(index));
57490         ms.get("desc").setDisabled(!cm.isSortable(index));
57491         if(this.grid.enableColLock !== false){
57492             ms.get("lock").setDisabled(cm.isLocked(index));
57493             ms.get("unlock").setDisabled(!cm.isLocked(index));
57494         }
57495         this.hmenu.show(hd, "tl-bl");
57496     },
57497
57498     handleHdOver : function(e){
57499         var hd = this.findHeaderCell(e.getTarget());
57500         if(hd && !this.headersDisabled){
57501             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57502                this.fly(hd).addClass("x-grid-hd-over");
57503             }
57504         }
57505     },
57506
57507     handleHdOut : function(e){
57508         var hd = this.findHeaderCell(e.getTarget());
57509         if(hd){
57510             this.fly(hd).removeClass("x-grid-hd-over");
57511         }
57512     },
57513
57514     handleSplitDblClick : function(e, t){
57515         var i = this.getCellIndex(t);
57516         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57517             this.autoSizeColumn(i, true);
57518             this.layout();
57519         }
57520     },
57521
57522     render : function(){
57523
57524         var cm = this.cm;
57525         var colCount = cm.getColumnCount();
57526
57527         if(this.grid.monitorWindowResize === true){
57528             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57529         }
57530         var header = this.renderHeaders();
57531         var body = this.templates.body.apply({rows:""});
57532         var html = this.templates.master.apply({
57533             lockedBody: body,
57534             body: body,
57535             lockedHeader: header[0],
57536             header: header[1]
57537         });
57538
57539         //this.updateColumns();
57540
57541         this.grid.getGridEl().dom.innerHTML = html;
57542
57543         this.initElements();
57544         
57545         // a kludge to fix the random scolling effect in webkit
57546         this.el.on("scroll", function() {
57547             this.el.dom.scrollTop=0; // hopefully not recursive..
57548         },this);
57549
57550         this.scroller.on("scroll", this.handleScroll, this);
57551         this.lockedBody.on("mousewheel", this.handleWheel, this);
57552         this.mainBody.on("mousewheel", this.handleWheel, this);
57553
57554         this.mainHd.on("mouseover", this.handleHdOver, this);
57555         this.mainHd.on("mouseout", this.handleHdOut, this);
57556         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57557                 {delegate: "."+this.splitClass});
57558
57559         this.lockedHd.on("mouseover", this.handleHdOver, this);
57560         this.lockedHd.on("mouseout", this.handleHdOut, this);
57561         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57562                 {delegate: "."+this.splitClass});
57563
57564         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57565             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57566         }
57567
57568         this.updateSplitters();
57569
57570         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57571             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57572             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57573         }
57574
57575         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57576             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57577             this.hmenu.add(
57578                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57579                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57580             );
57581             if(this.grid.enableColLock !== false){
57582                 this.hmenu.add('-',
57583                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57584                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57585                 );
57586             }
57587             if (Roo.isTouch) {
57588                  this.hmenu.add('-',
57589                     {id:"wider", text: this.columnsWiderText},
57590                     {id:"narrow", text: this.columnsNarrowText }
57591                 );
57592                 
57593                  
57594             }
57595             
57596             if(this.grid.enableColumnHide !== false){
57597
57598                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57599                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57600                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57601
57602                 this.hmenu.add('-',
57603                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57604                 );
57605             }
57606             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57607
57608             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57609         }
57610
57611         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57612             this.dd = new Roo.grid.GridDragZone(this.grid, {
57613                 ddGroup : this.grid.ddGroup || 'GridDD'
57614             });
57615             
57616         }
57617
57618         /*
57619         for(var i = 0; i < colCount; i++){
57620             if(cm.isHidden(i)){
57621                 this.hideColumn(i);
57622             }
57623             if(cm.config[i].align){
57624                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57625                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57626             }
57627         }*/
57628         
57629         this.updateHeaderSortState();
57630
57631         this.beforeInitialResize();
57632         this.layout(true);
57633
57634         // two part rendering gives faster view to the user
57635         this.renderPhase2.defer(1, this);
57636     },
57637
57638     renderPhase2 : function(){
57639         // render the rows now
57640         this.refresh();
57641         if(this.grid.autoSizeColumns){
57642             this.autoSizeColumns();
57643         }
57644     },
57645
57646     beforeInitialResize : function(){
57647
57648     },
57649
57650     onColumnSplitterMoved : function(i, w){
57651         this.userResized = true;
57652         var cm = this.grid.colModel;
57653         cm.setColumnWidth(i, w, true);
57654         var cid = cm.getColumnId(i);
57655         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57656         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57657         this.updateSplitters();
57658         this.layout();
57659         this.grid.fireEvent("columnresize", i, w);
57660     },
57661
57662     syncRowHeights : function(startIndex, endIndex){
57663         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57664             startIndex = startIndex || 0;
57665             var mrows = this.getBodyTable().rows;
57666             var lrows = this.getLockedTable().rows;
57667             var len = mrows.length-1;
57668             endIndex = Math.min(endIndex || len, len);
57669             for(var i = startIndex; i <= endIndex; i++){
57670                 var m = mrows[i], l = lrows[i];
57671                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57672                 m.style.height = l.style.height = h + "px";
57673             }
57674         }
57675     },
57676
57677     layout : function(initialRender, is2ndPass){
57678         var g = this.grid;
57679         var auto = g.autoHeight;
57680         var scrollOffset = 16;
57681         var c = g.getGridEl(), cm = this.cm,
57682                 expandCol = g.autoExpandColumn,
57683                 gv = this;
57684         //c.beginMeasure();
57685
57686         if(!c.dom.offsetWidth){ // display:none?
57687             if(initialRender){
57688                 this.lockedWrap.show();
57689                 this.mainWrap.show();
57690             }
57691             return;
57692         }
57693
57694         var hasLock = this.cm.isLocked(0);
57695
57696         var tbh = this.headerPanel.getHeight();
57697         var bbh = this.footerPanel.getHeight();
57698
57699         if(auto){
57700             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57701             var newHeight = ch + c.getBorderWidth("tb");
57702             if(g.maxHeight){
57703                 newHeight = Math.min(g.maxHeight, newHeight);
57704             }
57705             c.setHeight(newHeight);
57706         }
57707
57708         if(g.autoWidth){
57709             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57710         }
57711
57712         var s = this.scroller;
57713
57714         var csize = c.getSize(true);
57715
57716         this.el.setSize(csize.width, csize.height);
57717
57718         this.headerPanel.setWidth(csize.width);
57719         this.footerPanel.setWidth(csize.width);
57720
57721         var hdHeight = this.mainHd.getHeight();
57722         var vw = csize.width;
57723         var vh = csize.height - (tbh + bbh);
57724
57725         s.setSize(vw, vh);
57726
57727         var bt = this.getBodyTable();
57728         
57729         if(cm.getLockedCount() == cm.config.length){
57730             bt = this.getLockedTable();
57731         }
57732         
57733         var ltWidth = hasLock ?
57734                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57735
57736         var scrollHeight = bt.offsetHeight;
57737         var scrollWidth = ltWidth + bt.offsetWidth;
57738         var vscroll = false, hscroll = false;
57739
57740         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57741
57742         var lw = this.lockedWrap, mw = this.mainWrap;
57743         var lb = this.lockedBody, mb = this.mainBody;
57744
57745         setTimeout(function(){
57746             var t = s.dom.offsetTop;
57747             var w = s.dom.clientWidth,
57748                 h = s.dom.clientHeight;
57749
57750             lw.setTop(t);
57751             lw.setSize(ltWidth, h);
57752
57753             mw.setLeftTop(ltWidth, t);
57754             mw.setSize(w-ltWidth, h);
57755
57756             lb.setHeight(h-hdHeight);
57757             mb.setHeight(h-hdHeight);
57758
57759             if(is2ndPass !== true && !gv.userResized && expandCol){
57760                 // high speed resize without full column calculation
57761                 
57762                 var ci = cm.getIndexById(expandCol);
57763                 if (ci < 0) {
57764                     ci = cm.findColumnIndex(expandCol);
57765                 }
57766                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57767                 var expandId = cm.getColumnId(ci);
57768                 var  tw = cm.getTotalWidth(false);
57769                 var currentWidth = cm.getColumnWidth(ci);
57770                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57771                 if(currentWidth != cw){
57772                     cm.setColumnWidth(ci, cw, true);
57773                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57774                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57775                     gv.updateSplitters();
57776                     gv.layout(false, true);
57777                 }
57778             }
57779
57780             if(initialRender){
57781                 lw.show();
57782                 mw.show();
57783             }
57784             //c.endMeasure();
57785         }, 10);
57786     },
57787
57788     onWindowResize : function(){
57789         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57790             return;
57791         }
57792         this.layout();
57793     },
57794
57795     appendFooter : function(parentEl){
57796         return null;
57797     },
57798
57799     sortAscText : "Sort Ascending",
57800     sortDescText : "Sort Descending",
57801     lockText : "Lock Column",
57802     unlockText : "Unlock Column",
57803     columnsText : "Columns",
57804  
57805     columnsWiderText : "Wider",
57806     columnsNarrowText : "Thinner"
57807 });
57808
57809
57810 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57811     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57812     this.proxy.el.addClass('x-grid3-col-dd');
57813 };
57814
57815 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57816     handleMouseDown : function(e){
57817
57818     },
57819
57820     callHandleMouseDown : function(e){
57821         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57822     }
57823 });
57824 /*
57825  * Based on:
57826  * Ext JS Library 1.1.1
57827  * Copyright(c) 2006-2007, Ext JS, LLC.
57828  *
57829  * Originally Released Under LGPL - original licence link has changed is not relivant.
57830  *
57831  * Fork - LGPL
57832  * <script type="text/javascript">
57833  */
57834  
57835 // private
57836 // This is a support class used internally by the Grid components
57837 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57838     this.grid = grid;
57839     this.view = grid.getView();
57840     this.proxy = this.view.resizeProxy;
57841     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57842         "gridSplitters" + this.grid.getGridEl().id, {
57843         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57844     });
57845     this.setHandleElId(Roo.id(hd));
57846     this.setOuterHandleElId(Roo.id(hd2));
57847     this.scroll = false;
57848 };
57849 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57850     fly: Roo.Element.fly,
57851
57852     b4StartDrag : function(x, y){
57853         this.view.headersDisabled = true;
57854         this.proxy.setHeight(this.view.mainWrap.getHeight());
57855         var w = this.cm.getColumnWidth(this.cellIndex);
57856         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57857         this.resetConstraints();
57858         this.setXConstraint(minw, 1000);
57859         this.setYConstraint(0, 0);
57860         this.minX = x - minw;
57861         this.maxX = x + 1000;
57862         this.startPos = x;
57863         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57864     },
57865
57866
57867     handleMouseDown : function(e){
57868         ev = Roo.EventObject.setEvent(e);
57869         var t = this.fly(ev.getTarget());
57870         if(t.hasClass("x-grid-split")){
57871             this.cellIndex = this.view.getCellIndex(t.dom);
57872             this.split = t.dom;
57873             this.cm = this.grid.colModel;
57874             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57875                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57876             }
57877         }
57878     },
57879
57880     endDrag : function(e){
57881         this.view.headersDisabled = false;
57882         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57883         var diff = endX - this.startPos;
57884         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57885     },
57886
57887     autoOffset : function(){
57888         this.setDelta(0,0);
57889     }
57890 });/*
57891  * Based on:
57892  * Ext JS Library 1.1.1
57893  * Copyright(c) 2006-2007, Ext JS, LLC.
57894  *
57895  * Originally Released Under LGPL - original licence link has changed is not relivant.
57896  *
57897  * Fork - LGPL
57898  * <script type="text/javascript">
57899  */
57900  
57901 // private
57902 // This is a support class used internally by the Grid components
57903 Roo.grid.GridDragZone = function(grid, config){
57904     this.view = grid.getView();
57905     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57906     if(this.view.lockedBody){
57907         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57908         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57909     }
57910     this.scroll = false;
57911     this.grid = grid;
57912     this.ddel = document.createElement('div');
57913     this.ddel.className = 'x-grid-dd-wrap';
57914 };
57915
57916 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57917     ddGroup : "GridDD",
57918
57919     getDragData : function(e){
57920         var t = Roo.lib.Event.getTarget(e);
57921         var rowIndex = this.view.findRowIndex(t);
57922         var sm = this.grid.selModel;
57923             
57924         //Roo.log(rowIndex);
57925         
57926         if (sm.getSelectedCell) {
57927             // cell selection..
57928             if (!sm.getSelectedCell()) {
57929                 return false;
57930             }
57931             if (rowIndex != sm.getSelectedCell()[0]) {
57932                 return false;
57933             }
57934         
57935         }
57936         
57937         if(rowIndex !== false){
57938             
57939             // if editorgrid.. 
57940             
57941             
57942             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57943                
57944             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57945               //  
57946             //}
57947             if (e.hasModifier()){
57948                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57949             }
57950             
57951             Roo.log("getDragData");
57952             
57953             return {
57954                 grid: this.grid,
57955                 ddel: this.ddel,
57956                 rowIndex: rowIndex,
57957                 selections:sm.getSelections ? sm.getSelections() : (
57958                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57959                 )
57960             };
57961         }
57962         return false;
57963     },
57964
57965     onInitDrag : function(e){
57966         var data = this.dragData;
57967         this.ddel.innerHTML = this.grid.getDragDropText();
57968         this.proxy.update(this.ddel);
57969         // fire start drag?
57970     },
57971
57972     afterRepair : function(){
57973         this.dragging = false;
57974     },
57975
57976     getRepairXY : function(e, data){
57977         return false;
57978     },
57979
57980     onEndDrag : function(data, e){
57981         // fire end drag?
57982     },
57983
57984     onValidDrop : function(dd, e, id){
57985         // fire drag drop?
57986         this.hideProxy();
57987     },
57988
57989     beforeInvalidDrop : function(e, id){
57990
57991     }
57992 });/*
57993  * Based on:
57994  * Ext JS Library 1.1.1
57995  * Copyright(c) 2006-2007, Ext JS, LLC.
57996  *
57997  * Originally Released Under LGPL - original licence link has changed is not relivant.
57998  *
57999  * Fork - LGPL
58000  * <script type="text/javascript">
58001  */
58002  
58003
58004 /**
58005  * @class Roo.grid.ColumnModel
58006  * @extends Roo.util.Observable
58007  * This is the default implementation of a ColumnModel used by the Grid. It defines
58008  * the columns in the grid.
58009  * <br>Usage:<br>
58010  <pre><code>
58011  var colModel = new Roo.grid.ColumnModel([
58012         {header: "Ticker", width: 60, sortable: true, locked: true},
58013         {header: "Company Name", width: 150, sortable: true},
58014         {header: "Market Cap.", width: 100, sortable: true},
58015         {header: "$ Sales", width: 100, sortable: true, renderer: money},
58016         {header: "Employees", width: 100, sortable: true, resizable: false}
58017  ]);
58018  </code></pre>
58019  * <p>
58020  
58021  * The config options listed for this class are options which may appear in each
58022  * individual column definition.
58023  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
58024  * @constructor
58025  * @param {Object} config An Array of column config objects. See this class's
58026  * config objects for details.
58027 */
58028 Roo.grid.ColumnModel = function(config){
58029         /**
58030      * The config passed into the constructor
58031      */
58032     this.config = config;
58033     this.lookup = {};
58034
58035     // if no id, create one
58036     // if the column does not have a dataIndex mapping,
58037     // map it to the order it is in the config
58038     for(var i = 0, len = config.length; i < len; i++){
58039         var c = config[i];
58040         if(typeof c.dataIndex == "undefined"){
58041             c.dataIndex = i;
58042         }
58043         if(typeof c.renderer == "string"){
58044             c.renderer = Roo.util.Format[c.renderer];
58045         }
58046         if(typeof c.id == "undefined"){
58047             c.id = Roo.id();
58048         }
58049         if(c.editor && c.editor.xtype){
58050             c.editor  = Roo.factory(c.editor, Roo.grid);
58051         }
58052         if(c.editor && c.editor.isFormField){
58053             c.editor = new Roo.grid.GridEditor(c.editor);
58054         }
58055         this.lookup[c.id] = c;
58056     }
58057
58058     /**
58059      * The width of columns which have no width specified (defaults to 100)
58060      * @type Number
58061      */
58062     this.defaultWidth = 100;
58063
58064     /**
58065      * Default sortable of columns which have no sortable specified (defaults to false)
58066      * @type Boolean
58067      */
58068     this.defaultSortable = false;
58069
58070     this.addEvents({
58071         /**
58072              * @event widthchange
58073              * Fires when the width of a column changes.
58074              * @param {ColumnModel} this
58075              * @param {Number} columnIndex The column index
58076              * @param {Number} newWidth The new width
58077              */
58078             "widthchange": true,
58079         /**
58080              * @event headerchange
58081              * Fires when the text of a header changes.
58082              * @param {ColumnModel} this
58083              * @param {Number} columnIndex The column index
58084              * @param {Number} newText The new header text
58085              */
58086             "headerchange": true,
58087         /**
58088              * @event hiddenchange
58089              * Fires when a column is hidden or "unhidden".
58090              * @param {ColumnModel} this
58091              * @param {Number} columnIndex The column index
58092              * @param {Boolean} hidden true if hidden, false otherwise
58093              */
58094             "hiddenchange": true,
58095             /**
58096          * @event columnmoved
58097          * Fires when a column is moved.
58098          * @param {ColumnModel} this
58099          * @param {Number} oldIndex
58100          * @param {Number} newIndex
58101          */
58102         "columnmoved" : true,
58103         /**
58104          * @event columlockchange
58105          * Fires when a column's locked state is changed
58106          * @param {ColumnModel} this
58107          * @param {Number} colIndex
58108          * @param {Boolean} locked true if locked
58109          */
58110         "columnlockchange" : true
58111     });
58112     Roo.grid.ColumnModel.superclass.constructor.call(this);
58113 };
58114 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
58115     /**
58116      * @cfg {String} header The header text to display in the Grid view.
58117      */
58118     /**
58119      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
58120      * {@link Roo.data.Record} definition from which to draw the column's value. If not
58121      * specified, the column's index is used as an index into the Record's data Array.
58122      */
58123     /**
58124      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
58125      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
58126      */
58127     /**
58128      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
58129      * Defaults to the value of the {@link #defaultSortable} property.
58130      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
58131      */
58132     /**
58133      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
58134      */
58135     /**
58136      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
58137      */
58138     /**
58139      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
58140      */
58141     /**
58142      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
58143      */
58144     /**
58145      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
58146      * given the cell's data value. See {@link #setRenderer}. If not specified, the
58147      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
58148      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
58149      */
58150        /**
58151      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
58152      */
58153     /**
58154      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
58155      */
58156     /**
58157      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
58158      */
58159     /**
58160      * @cfg {String} cursor (Optional)
58161      */
58162     /**
58163      * @cfg {String} tooltip (Optional)
58164      */
58165     /**
58166      * @cfg {Number} xs (Optional)
58167      */
58168     /**
58169      * @cfg {Number} sm (Optional)
58170      */
58171     /**
58172      * @cfg {Number} md (Optional)
58173      */
58174     /**
58175      * @cfg {Number} lg (Optional)
58176      */
58177     /**
58178      * Returns the id of the column at the specified index.
58179      * @param {Number} index The column index
58180      * @return {String} the id
58181      */
58182     getColumnId : function(index){
58183         return this.config[index].id;
58184     },
58185
58186     /**
58187      * Returns the column for a specified id.
58188      * @param {String} id The column id
58189      * @return {Object} the column
58190      */
58191     getColumnById : function(id){
58192         return this.lookup[id];
58193     },
58194
58195     
58196     /**
58197      * Returns the column for a specified dataIndex.
58198      * @param {String} dataIndex The column dataIndex
58199      * @return {Object|Boolean} the column or false if not found
58200      */
58201     getColumnByDataIndex: function(dataIndex){
58202         var index = this.findColumnIndex(dataIndex);
58203         return index > -1 ? this.config[index] : false;
58204     },
58205     
58206     /**
58207      * Returns the index for a specified column id.
58208      * @param {String} id The column id
58209      * @return {Number} the index, or -1 if not found
58210      */
58211     getIndexById : function(id){
58212         for(var i = 0, len = this.config.length; i < len; i++){
58213             if(this.config[i].id == id){
58214                 return i;
58215             }
58216         }
58217         return -1;
58218     },
58219     
58220     /**
58221      * Returns the index for a specified column dataIndex.
58222      * @param {String} dataIndex The column dataIndex
58223      * @return {Number} the index, or -1 if not found
58224      */
58225     
58226     findColumnIndex : function(dataIndex){
58227         for(var i = 0, len = this.config.length; i < len; i++){
58228             if(this.config[i].dataIndex == dataIndex){
58229                 return i;
58230             }
58231         }
58232         return -1;
58233     },
58234     
58235     
58236     moveColumn : function(oldIndex, newIndex){
58237         var c = this.config[oldIndex];
58238         this.config.splice(oldIndex, 1);
58239         this.config.splice(newIndex, 0, c);
58240         this.dataMap = null;
58241         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58242     },
58243
58244     isLocked : function(colIndex){
58245         return this.config[colIndex].locked === true;
58246     },
58247
58248     setLocked : function(colIndex, value, suppressEvent){
58249         if(this.isLocked(colIndex) == value){
58250             return;
58251         }
58252         this.config[colIndex].locked = value;
58253         if(!suppressEvent){
58254             this.fireEvent("columnlockchange", this, colIndex, value);
58255         }
58256     },
58257
58258     getTotalLockedWidth : function(){
58259         var totalWidth = 0;
58260         for(var i = 0; i < this.config.length; i++){
58261             if(this.isLocked(i) && !this.isHidden(i)){
58262                 this.totalWidth += this.getColumnWidth(i);
58263             }
58264         }
58265         return totalWidth;
58266     },
58267
58268     getLockedCount : function(){
58269         for(var i = 0, len = this.config.length; i < len; i++){
58270             if(!this.isLocked(i)){
58271                 return i;
58272             }
58273         }
58274         
58275         return this.config.length;
58276     },
58277
58278     /**
58279      * Returns the number of columns.
58280      * @return {Number}
58281      */
58282     getColumnCount : function(visibleOnly){
58283         if(visibleOnly === true){
58284             var c = 0;
58285             for(var i = 0, len = this.config.length; i < len; i++){
58286                 if(!this.isHidden(i)){
58287                     c++;
58288                 }
58289             }
58290             return c;
58291         }
58292         return this.config.length;
58293     },
58294
58295     /**
58296      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58297      * @param {Function} fn
58298      * @param {Object} scope (optional)
58299      * @return {Array} result
58300      */
58301     getColumnsBy : function(fn, scope){
58302         var r = [];
58303         for(var i = 0, len = this.config.length; i < len; i++){
58304             var c = this.config[i];
58305             if(fn.call(scope||this, c, i) === true){
58306                 r[r.length] = c;
58307             }
58308         }
58309         return r;
58310     },
58311
58312     /**
58313      * Returns true if the specified column is sortable.
58314      * @param {Number} col The column index
58315      * @return {Boolean}
58316      */
58317     isSortable : function(col){
58318         if(typeof this.config[col].sortable == "undefined"){
58319             return this.defaultSortable;
58320         }
58321         return this.config[col].sortable;
58322     },
58323
58324     /**
58325      * Returns the rendering (formatting) function defined for the column.
58326      * @param {Number} col The column index.
58327      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58328      */
58329     getRenderer : function(col){
58330         if(!this.config[col].renderer){
58331             return Roo.grid.ColumnModel.defaultRenderer;
58332         }
58333         return this.config[col].renderer;
58334     },
58335
58336     /**
58337      * Sets the rendering (formatting) function for a column.
58338      * @param {Number} col The column index
58339      * @param {Function} fn The function to use to process the cell's raw data
58340      * to return HTML markup for the grid view. The render function is called with
58341      * the following parameters:<ul>
58342      * <li>Data value.</li>
58343      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58344      * <li>css A CSS style string to apply to the table cell.</li>
58345      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58346      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58347      * <li>Row index</li>
58348      * <li>Column index</li>
58349      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58350      */
58351     setRenderer : function(col, fn){
58352         this.config[col].renderer = fn;
58353     },
58354
58355     /**
58356      * Returns the width for the specified column.
58357      * @param {Number} col The column index
58358      * @return {Number}
58359      */
58360     getColumnWidth : function(col){
58361         return this.config[col].width * 1 || this.defaultWidth;
58362     },
58363
58364     /**
58365      * Sets the width for a column.
58366      * @param {Number} col The column index
58367      * @param {Number} width The new width
58368      */
58369     setColumnWidth : function(col, width, suppressEvent){
58370         this.config[col].width = width;
58371         this.totalWidth = null;
58372         if(!suppressEvent){
58373              this.fireEvent("widthchange", this, col, width);
58374         }
58375     },
58376
58377     /**
58378      * Returns the total width of all columns.
58379      * @param {Boolean} includeHidden True to include hidden column widths
58380      * @return {Number}
58381      */
58382     getTotalWidth : function(includeHidden){
58383         if(!this.totalWidth){
58384             this.totalWidth = 0;
58385             for(var i = 0, len = this.config.length; i < len; i++){
58386                 if(includeHidden || !this.isHidden(i)){
58387                     this.totalWidth += this.getColumnWidth(i);
58388                 }
58389             }
58390         }
58391         return this.totalWidth;
58392     },
58393
58394     /**
58395      * Returns the header for the specified column.
58396      * @param {Number} col The column index
58397      * @return {String}
58398      */
58399     getColumnHeader : function(col){
58400         return this.config[col].header;
58401     },
58402
58403     /**
58404      * Sets the header for a column.
58405      * @param {Number} col The column index
58406      * @param {String} header The new header
58407      */
58408     setColumnHeader : function(col, header){
58409         this.config[col].header = header;
58410         this.fireEvent("headerchange", this, col, header);
58411     },
58412
58413     /**
58414      * Returns the tooltip for the specified column.
58415      * @param {Number} col The column index
58416      * @return {String}
58417      */
58418     getColumnTooltip : function(col){
58419             return this.config[col].tooltip;
58420     },
58421     /**
58422      * Sets the tooltip for a column.
58423      * @param {Number} col The column index
58424      * @param {String} tooltip The new tooltip
58425      */
58426     setColumnTooltip : function(col, tooltip){
58427             this.config[col].tooltip = tooltip;
58428     },
58429
58430     /**
58431      * Returns the dataIndex for the specified column.
58432      * @param {Number} col The column index
58433      * @return {Number}
58434      */
58435     getDataIndex : function(col){
58436         return this.config[col].dataIndex;
58437     },
58438
58439     /**
58440      * Sets the dataIndex for a column.
58441      * @param {Number} col The column index
58442      * @param {Number} dataIndex The new dataIndex
58443      */
58444     setDataIndex : function(col, dataIndex){
58445         this.config[col].dataIndex = dataIndex;
58446     },
58447
58448     
58449     
58450     /**
58451      * Returns true if the cell is editable.
58452      * @param {Number} colIndex The column index
58453      * @param {Number} rowIndex The row index - this is nto actually used..?
58454      * @return {Boolean}
58455      */
58456     isCellEditable : function(colIndex, rowIndex){
58457         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58458     },
58459
58460     /**
58461      * Returns the editor defined for the cell/column.
58462      * return false or null to disable editing.
58463      * @param {Number} colIndex The column index
58464      * @param {Number} rowIndex The row index
58465      * @return {Object}
58466      */
58467     getCellEditor : function(colIndex, rowIndex){
58468         return this.config[colIndex].editor;
58469     },
58470
58471     /**
58472      * Sets if a column is editable.
58473      * @param {Number} col The column index
58474      * @param {Boolean} editable True if the column is editable
58475      */
58476     setEditable : function(col, editable){
58477         this.config[col].editable = editable;
58478     },
58479
58480
58481     /**
58482      * Returns true if the column is hidden.
58483      * @param {Number} colIndex The column index
58484      * @return {Boolean}
58485      */
58486     isHidden : function(colIndex){
58487         return this.config[colIndex].hidden;
58488     },
58489
58490
58491     /**
58492      * Returns true if the column width cannot be changed
58493      */
58494     isFixed : function(colIndex){
58495         return this.config[colIndex].fixed;
58496     },
58497
58498     /**
58499      * Returns true if the column can be resized
58500      * @return {Boolean}
58501      */
58502     isResizable : function(colIndex){
58503         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58504     },
58505     /**
58506      * Sets if a column is hidden.
58507      * @param {Number} colIndex The column index
58508      * @param {Boolean} hidden True if the column is hidden
58509      */
58510     setHidden : function(colIndex, hidden){
58511         this.config[colIndex].hidden = hidden;
58512         this.totalWidth = null;
58513         this.fireEvent("hiddenchange", this, colIndex, hidden);
58514     },
58515
58516     /**
58517      * Sets the editor for a column.
58518      * @param {Number} col The column index
58519      * @param {Object} editor The editor object
58520      */
58521     setEditor : function(col, editor){
58522         this.config[col].editor = editor;
58523     }
58524 });
58525
58526 Roo.grid.ColumnModel.defaultRenderer = function(value)
58527 {
58528     if(typeof value == "object") {
58529         return value;
58530     }
58531         if(typeof value == "string" && value.length < 1){
58532             return "&#160;";
58533         }
58534     
58535         return String.format("{0}", value);
58536 };
58537
58538 // Alias for backwards compatibility
58539 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58540 /*
58541  * Based on:
58542  * Ext JS Library 1.1.1
58543  * Copyright(c) 2006-2007, Ext JS, LLC.
58544  *
58545  * Originally Released Under LGPL - original licence link has changed is not relivant.
58546  *
58547  * Fork - LGPL
58548  * <script type="text/javascript">
58549  */
58550
58551 /**
58552  * @class Roo.grid.AbstractSelectionModel
58553  * @extends Roo.util.Observable
58554  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58555  * implemented by descendant classes.  This class should not be directly instantiated.
58556  * @constructor
58557  */
58558 Roo.grid.AbstractSelectionModel = function(){
58559     this.locked = false;
58560     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58561 };
58562
58563 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58564     /** @ignore Called by the grid automatically. Do not call directly. */
58565     init : function(grid){
58566         this.grid = grid;
58567         this.initEvents();
58568     },
58569
58570     /**
58571      * Locks the selections.
58572      */
58573     lock : function(){
58574         this.locked = true;
58575     },
58576
58577     /**
58578      * Unlocks the selections.
58579      */
58580     unlock : function(){
58581         this.locked = false;
58582     },
58583
58584     /**
58585      * Returns true if the selections are locked.
58586      * @return {Boolean}
58587      */
58588     isLocked : function(){
58589         return this.locked;
58590     }
58591 });/*
58592  * Based on:
58593  * Ext JS Library 1.1.1
58594  * Copyright(c) 2006-2007, Ext JS, LLC.
58595  *
58596  * Originally Released Under LGPL - original licence link has changed is not relivant.
58597  *
58598  * Fork - LGPL
58599  * <script type="text/javascript">
58600  */
58601 /**
58602  * @extends Roo.grid.AbstractSelectionModel
58603  * @class Roo.grid.RowSelectionModel
58604  * The default SelectionModel used by {@link Roo.grid.Grid}.
58605  * It supports multiple selections and keyboard selection/navigation. 
58606  * @constructor
58607  * @param {Object} config
58608  */
58609 Roo.grid.RowSelectionModel = function(config){
58610     Roo.apply(this, config);
58611     this.selections = new Roo.util.MixedCollection(false, function(o){
58612         return o.id;
58613     });
58614
58615     this.last = false;
58616     this.lastActive = false;
58617
58618     this.addEvents({
58619         /**
58620              * @event selectionchange
58621              * Fires when the selection changes
58622              * @param {SelectionModel} this
58623              */
58624             "selectionchange" : true,
58625         /**
58626              * @event afterselectionchange
58627              * Fires after the selection changes (eg. by key press or clicking)
58628              * @param {SelectionModel} this
58629              */
58630             "afterselectionchange" : true,
58631         /**
58632              * @event beforerowselect
58633              * Fires when a row is selected being selected, return false to cancel.
58634              * @param {SelectionModel} this
58635              * @param {Number} rowIndex The selected index
58636              * @param {Boolean} keepExisting False if other selections will be cleared
58637              */
58638             "beforerowselect" : true,
58639         /**
58640              * @event rowselect
58641              * Fires when a row is selected.
58642              * @param {SelectionModel} this
58643              * @param {Number} rowIndex The selected index
58644              * @param {Roo.data.Record} r The record
58645              */
58646             "rowselect" : true,
58647         /**
58648              * @event rowdeselect
58649              * Fires when a row is deselected.
58650              * @param {SelectionModel} this
58651              * @param {Number} rowIndex The selected index
58652              */
58653         "rowdeselect" : true
58654     });
58655     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58656     this.locked = false;
58657 };
58658
58659 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58660     /**
58661      * @cfg {Boolean} singleSelect
58662      * True to allow selection of only one row at a time (defaults to false)
58663      */
58664     singleSelect : false,
58665
58666     // private
58667     initEvents : function(){
58668
58669         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58670             this.grid.on("mousedown", this.handleMouseDown, this);
58671         }else{ // allow click to work like normal
58672             this.grid.on("rowclick", this.handleDragableRowClick, this);
58673         }
58674
58675         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58676             "up" : function(e){
58677                 if(!e.shiftKey){
58678                     this.selectPrevious(e.shiftKey);
58679                 }else if(this.last !== false && this.lastActive !== false){
58680                     var last = this.last;
58681                     this.selectRange(this.last,  this.lastActive-1);
58682                     this.grid.getView().focusRow(this.lastActive);
58683                     if(last !== false){
58684                         this.last = last;
58685                     }
58686                 }else{
58687                     this.selectFirstRow();
58688                 }
58689                 this.fireEvent("afterselectionchange", this);
58690             },
58691             "down" : function(e){
58692                 if(!e.shiftKey){
58693                     this.selectNext(e.shiftKey);
58694                 }else if(this.last !== false && this.lastActive !== false){
58695                     var last = this.last;
58696                     this.selectRange(this.last,  this.lastActive+1);
58697                     this.grid.getView().focusRow(this.lastActive);
58698                     if(last !== false){
58699                         this.last = last;
58700                     }
58701                 }else{
58702                     this.selectFirstRow();
58703                 }
58704                 this.fireEvent("afterselectionchange", this);
58705             },
58706             scope: this
58707         });
58708
58709         var view = this.grid.view;
58710         view.on("refresh", this.onRefresh, this);
58711         view.on("rowupdated", this.onRowUpdated, this);
58712         view.on("rowremoved", this.onRemove, this);
58713     },
58714
58715     // private
58716     onRefresh : function(){
58717         var ds = this.grid.dataSource, i, v = this.grid.view;
58718         var s = this.selections;
58719         s.each(function(r){
58720             if((i = ds.indexOfId(r.id)) != -1){
58721                 v.onRowSelect(i);
58722                 s.add(ds.getAt(i)); // updating the selection relate data
58723             }else{
58724                 s.remove(r);
58725             }
58726         });
58727     },
58728
58729     // private
58730     onRemove : function(v, index, r){
58731         this.selections.remove(r);
58732     },
58733
58734     // private
58735     onRowUpdated : function(v, index, r){
58736         if(this.isSelected(r)){
58737             v.onRowSelect(index);
58738         }
58739     },
58740
58741     /**
58742      * Select records.
58743      * @param {Array} records The records to select
58744      * @param {Boolean} keepExisting (optional) True to keep existing selections
58745      */
58746     selectRecords : function(records, keepExisting){
58747         if(!keepExisting){
58748             this.clearSelections();
58749         }
58750         var ds = this.grid.dataSource;
58751         for(var i = 0, len = records.length; i < len; i++){
58752             this.selectRow(ds.indexOf(records[i]), true);
58753         }
58754     },
58755
58756     /**
58757      * Gets the number of selected rows.
58758      * @return {Number}
58759      */
58760     getCount : function(){
58761         return this.selections.length;
58762     },
58763
58764     /**
58765      * Selects the first row in the grid.
58766      */
58767     selectFirstRow : function(){
58768         this.selectRow(0);
58769     },
58770
58771     /**
58772      * Select the last row.
58773      * @param {Boolean} keepExisting (optional) True to keep existing selections
58774      */
58775     selectLastRow : function(keepExisting){
58776         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58777     },
58778
58779     /**
58780      * Selects the row immediately following the last selected row.
58781      * @param {Boolean} keepExisting (optional) True to keep existing selections
58782      */
58783     selectNext : function(keepExisting){
58784         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58785             this.selectRow(this.last+1, keepExisting);
58786             this.grid.getView().focusRow(this.last);
58787         }
58788     },
58789
58790     /**
58791      * Selects the row that precedes the last selected row.
58792      * @param {Boolean} keepExisting (optional) True to keep existing selections
58793      */
58794     selectPrevious : function(keepExisting){
58795         if(this.last){
58796             this.selectRow(this.last-1, keepExisting);
58797             this.grid.getView().focusRow(this.last);
58798         }
58799     },
58800
58801     /**
58802      * Returns the selected records
58803      * @return {Array} Array of selected records
58804      */
58805     getSelections : function(){
58806         return [].concat(this.selections.items);
58807     },
58808
58809     /**
58810      * Returns the first selected record.
58811      * @return {Record}
58812      */
58813     getSelected : function(){
58814         return this.selections.itemAt(0);
58815     },
58816
58817
58818     /**
58819      * Clears all selections.
58820      */
58821     clearSelections : function(fast){
58822         if(this.locked) {
58823             return;
58824         }
58825         if(fast !== true){
58826             var ds = this.grid.dataSource;
58827             var s = this.selections;
58828             s.each(function(r){
58829                 this.deselectRow(ds.indexOfId(r.id));
58830             }, this);
58831             s.clear();
58832         }else{
58833             this.selections.clear();
58834         }
58835         this.last = false;
58836     },
58837
58838
58839     /**
58840      * Selects all rows.
58841      */
58842     selectAll : function(){
58843         if(this.locked) {
58844             return;
58845         }
58846         this.selections.clear();
58847         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58848             this.selectRow(i, true);
58849         }
58850     },
58851
58852     /**
58853      * Returns True if there is a selection.
58854      * @return {Boolean}
58855      */
58856     hasSelection : function(){
58857         return this.selections.length > 0;
58858     },
58859
58860     /**
58861      * Returns True if the specified row is selected.
58862      * @param {Number/Record} record The record or index of the record to check
58863      * @return {Boolean}
58864      */
58865     isSelected : function(index){
58866         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58867         return (r && this.selections.key(r.id) ? true : false);
58868     },
58869
58870     /**
58871      * Returns True if the specified record id is selected.
58872      * @param {String} id The id of record to check
58873      * @return {Boolean}
58874      */
58875     isIdSelected : function(id){
58876         return (this.selections.key(id) ? true : false);
58877     },
58878
58879     // private
58880     handleMouseDown : function(e, t){
58881         var view = this.grid.getView(), rowIndex;
58882         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58883             return;
58884         };
58885         if(e.shiftKey && this.last !== false){
58886             var last = this.last;
58887             this.selectRange(last, rowIndex, e.ctrlKey);
58888             this.last = last; // reset the last
58889             view.focusRow(rowIndex);
58890         }else{
58891             var isSelected = this.isSelected(rowIndex);
58892             if(e.button !== 0 && isSelected){
58893                 view.focusRow(rowIndex);
58894             }else if(e.ctrlKey && isSelected){
58895                 this.deselectRow(rowIndex);
58896             }else if(!isSelected){
58897                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58898                 view.focusRow(rowIndex);
58899             }
58900         }
58901         this.fireEvent("afterselectionchange", this);
58902     },
58903     // private
58904     handleDragableRowClick :  function(grid, rowIndex, e) 
58905     {
58906         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58907             this.selectRow(rowIndex, false);
58908             grid.view.focusRow(rowIndex);
58909              this.fireEvent("afterselectionchange", this);
58910         }
58911     },
58912     
58913     /**
58914      * Selects multiple rows.
58915      * @param {Array} rows Array of the indexes of the row to select
58916      * @param {Boolean} keepExisting (optional) True to keep existing selections
58917      */
58918     selectRows : function(rows, keepExisting){
58919         if(!keepExisting){
58920             this.clearSelections();
58921         }
58922         for(var i = 0, len = rows.length; i < len; i++){
58923             this.selectRow(rows[i], true);
58924         }
58925     },
58926
58927     /**
58928      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58929      * @param {Number} startRow The index of the first row in the range
58930      * @param {Number} endRow The index of the last row in the range
58931      * @param {Boolean} keepExisting (optional) True to retain existing selections
58932      */
58933     selectRange : function(startRow, endRow, keepExisting){
58934         if(this.locked) {
58935             return;
58936         }
58937         if(!keepExisting){
58938             this.clearSelections();
58939         }
58940         if(startRow <= endRow){
58941             for(var i = startRow; i <= endRow; i++){
58942                 this.selectRow(i, true);
58943             }
58944         }else{
58945             for(var i = startRow; i >= endRow; i--){
58946                 this.selectRow(i, true);
58947             }
58948         }
58949     },
58950
58951     /**
58952      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58953      * @param {Number} startRow The index of the first row in the range
58954      * @param {Number} endRow The index of the last row in the range
58955      */
58956     deselectRange : function(startRow, endRow, preventViewNotify){
58957         if(this.locked) {
58958             return;
58959         }
58960         for(var i = startRow; i <= endRow; i++){
58961             this.deselectRow(i, preventViewNotify);
58962         }
58963     },
58964
58965     /**
58966      * Selects a row.
58967      * @param {Number} row The index of the row to select
58968      * @param {Boolean} keepExisting (optional) True to keep existing selections
58969      */
58970     selectRow : function(index, keepExisting, preventViewNotify){
58971         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58972             return;
58973         }
58974         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58975             if(!keepExisting || this.singleSelect){
58976                 this.clearSelections();
58977             }
58978             var r = this.grid.dataSource.getAt(index);
58979             this.selections.add(r);
58980             this.last = this.lastActive = index;
58981             if(!preventViewNotify){
58982                 this.grid.getView().onRowSelect(index);
58983             }
58984             this.fireEvent("rowselect", this, index, r);
58985             this.fireEvent("selectionchange", this);
58986         }
58987     },
58988
58989     /**
58990      * Deselects a row.
58991      * @param {Number} row The index of the row to deselect
58992      */
58993     deselectRow : function(index, preventViewNotify){
58994         if(this.locked) {
58995             return;
58996         }
58997         if(this.last == index){
58998             this.last = false;
58999         }
59000         if(this.lastActive == index){
59001             this.lastActive = false;
59002         }
59003         var r = this.grid.dataSource.getAt(index);
59004         this.selections.remove(r);
59005         if(!preventViewNotify){
59006             this.grid.getView().onRowDeselect(index);
59007         }
59008         this.fireEvent("rowdeselect", this, index);
59009         this.fireEvent("selectionchange", this);
59010     },
59011
59012     // private
59013     restoreLast : function(){
59014         if(this._last){
59015             this.last = this._last;
59016         }
59017     },
59018
59019     // private
59020     acceptsNav : function(row, col, cm){
59021         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59022     },
59023
59024     // private
59025     onEditorKey : function(field, e){
59026         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
59027         if(k == e.TAB){
59028             e.stopEvent();
59029             ed.completeEdit();
59030             if(e.shiftKey){
59031                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59032             }else{
59033                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59034             }
59035         }else if(k == e.ENTER && !e.ctrlKey){
59036             e.stopEvent();
59037             ed.completeEdit();
59038             if(e.shiftKey){
59039                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
59040             }else{
59041                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
59042             }
59043         }else if(k == e.ESC){
59044             ed.cancelEdit();
59045         }
59046         if(newCell){
59047             g.startEditing(newCell[0], newCell[1]);
59048         }
59049     }
59050 });/*
59051  * Based on:
59052  * Ext JS Library 1.1.1
59053  * Copyright(c) 2006-2007, Ext JS, LLC.
59054  *
59055  * Originally Released Under LGPL - original licence link has changed is not relivant.
59056  *
59057  * Fork - LGPL
59058  * <script type="text/javascript">
59059  */
59060 /**
59061  * @class Roo.grid.CellSelectionModel
59062  * @extends Roo.grid.AbstractSelectionModel
59063  * This class provides the basic implementation for cell selection in a grid.
59064  * @constructor
59065  * @param {Object} config The object containing the configuration of this model.
59066  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
59067  */
59068 Roo.grid.CellSelectionModel = function(config){
59069     Roo.apply(this, config);
59070
59071     this.selection = null;
59072
59073     this.addEvents({
59074         /**
59075              * @event beforerowselect
59076              * Fires before a cell is selected.
59077              * @param {SelectionModel} this
59078              * @param {Number} rowIndex The selected row index
59079              * @param {Number} colIndex The selected cell index
59080              */
59081             "beforecellselect" : true,
59082         /**
59083              * @event cellselect
59084              * Fires when a cell is selected.
59085              * @param {SelectionModel} this
59086              * @param {Number} rowIndex The selected row index
59087              * @param {Number} colIndex The selected cell index
59088              */
59089             "cellselect" : true,
59090         /**
59091              * @event selectionchange
59092              * Fires when the active selection changes.
59093              * @param {SelectionModel} this
59094              * @param {Object} selection null for no selection or an object (o) with two properties
59095                 <ul>
59096                 <li>o.record: the record object for the row the selection is in</li>
59097                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
59098                 </ul>
59099              */
59100             "selectionchange" : true,
59101         /**
59102              * @event tabend
59103              * Fires when the tab (or enter) was pressed on the last editable cell
59104              * You can use this to trigger add new row.
59105              * @param {SelectionModel} this
59106              */
59107             "tabend" : true,
59108          /**
59109              * @event beforeeditnext
59110              * Fires before the next editable sell is made active
59111              * You can use this to skip to another cell or fire the tabend
59112              *    if you set cell to false
59113              * @param {Object} eventdata object : { cell : [ row, col ] } 
59114              */
59115             "beforeeditnext" : true
59116     });
59117     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
59118 };
59119
59120 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
59121     
59122     enter_is_tab: false,
59123
59124     /** @ignore */
59125     initEvents : function(){
59126         this.grid.on("mousedown", this.handleMouseDown, this);
59127         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
59128         var view = this.grid.view;
59129         view.on("refresh", this.onViewChange, this);
59130         view.on("rowupdated", this.onRowUpdated, this);
59131         view.on("beforerowremoved", this.clearSelections, this);
59132         view.on("beforerowsinserted", this.clearSelections, this);
59133         if(this.grid.isEditor){
59134             this.grid.on("beforeedit", this.beforeEdit,  this);
59135         }
59136     },
59137
59138         //private
59139     beforeEdit : function(e){
59140         this.select(e.row, e.column, false, true, e.record);
59141     },
59142
59143         //private
59144     onRowUpdated : function(v, index, r){
59145         if(this.selection && this.selection.record == r){
59146             v.onCellSelect(index, this.selection.cell[1]);
59147         }
59148     },
59149
59150         //private
59151     onViewChange : function(){
59152         this.clearSelections(true);
59153     },
59154
59155         /**
59156          * Returns the currently selected cell,.
59157          * @return {Array} The selected cell (row, column) or null if none selected.
59158          */
59159     getSelectedCell : function(){
59160         return this.selection ? this.selection.cell : null;
59161     },
59162
59163     /**
59164      * Clears all selections.
59165      * @param {Boolean} true to prevent the gridview from being notified about the change.
59166      */
59167     clearSelections : function(preventNotify){
59168         var s = this.selection;
59169         if(s){
59170             if(preventNotify !== true){
59171                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
59172             }
59173             this.selection = null;
59174             this.fireEvent("selectionchange", this, null);
59175         }
59176     },
59177
59178     /**
59179      * Returns true if there is a selection.
59180      * @return {Boolean}
59181      */
59182     hasSelection : function(){
59183         return this.selection ? true : false;
59184     },
59185
59186     /** @ignore */
59187     handleMouseDown : function(e, t){
59188         var v = this.grid.getView();
59189         if(this.isLocked()){
59190             return;
59191         };
59192         var row = v.findRowIndex(t);
59193         var cell = v.findCellIndex(t);
59194         if(row !== false && cell !== false){
59195             this.select(row, cell);
59196         }
59197     },
59198
59199     /**
59200      * Selects a cell.
59201      * @param {Number} rowIndex
59202      * @param {Number} collIndex
59203      */
59204     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59205         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59206             this.clearSelections();
59207             r = r || this.grid.dataSource.getAt(rowIndex);
59208             this.selection = {
59209                 record : r,
59210                 cell : [rowIndex, colIndex]
59211             };
59212             if(!preventViewNotify){
59213                 var v = this.grid.getView();
59214                 v.onCellSelect(rowIndex, colIndex);
59215                 if(preventFocus !== true){
59216                     v.focusCell(rowIndex, colIndex);
59217                 }
59218             }
59219             this.fireEvent("cellselect", this, rowIndex, colIndex);
59220             this.fireEvent("selectionchange", this, this.selection);
59221         }
59222     },
59223
59224         //private
59225     isSelectable : function(rowIndex, colIndex, cm){
59226         return !cm.isHidden(colIndex);
59227     },
59228
59229     /** @ignore */
59230     handleKeyDown : function(e){
59231         //Roo.log('Cell Sel Model handleKeyDown');
59232         if(!e.isNavKeyPress()){
59233             return;
59234         }
59235         var g = this.grid, s = this.selection;
59236         if(!s){
59237             e.stopEvent();
59238             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59239             if(cell){
59240                 this.select(cell[0], cell[1]);
59241             }
59242             return;
59243         }
59244         var sm = this;
59245         var walk = function(row, col, step){
59246             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59247         };
59248         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59249         var newCell;
59250
59251       
59252
59253         switch(k){
59254             case e.TAB:
59255                 // handled by onEditorKey
59256                 if (g.isEditor && g.editing) {
59257                     return;
59258                 }
59259                 if(e.shiftKey) {
59260                     newCell = walk(r, c-1, -1);
59261                 } else {
59262                     newCell = walk(r, c+1, 1);
59263                 }
59264                 break;
59265             
59266             case e.DOWN:
59267                newCell = walk(r+1, c, 1);
59268                 break;
59269             
59270             case e.UP:
59271                 newCell = walk(r-1, c, -1);
59272                 break;
59273             
59274             case e.RIGHT:
59275                 newCell = walk(r, c+1, 1);
59276                 break;
59277             
59278             case e.LEFT:
59279                 newCell = walk(r, c-1, -1);
59280                 break;
59281             
59282             case e.ENTER:
59283                 
59284                 if(g.isEditor && !g.editing){
59285                    g.startEditing(r, c);
59286                    e.stopEvent();
59287                    return;
59288                 }
59289                 
59290                 
59291              break;
59292         };
59293         if(newCell){
59294             this.select(newCell[0], newCell[1]);
59295             e.stopEvent();
59296             
59297         }
59298     },
59299
59300     acceptsNav : function(row, col, cm){
59301         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59302     },
59303     /**
59304      * Selects a cell.
59305      * @param {Number} field (not used) - as it's normally used as a listener
59306      * @param {Number} e - event - fake it by using
59307      *
59308      * var e = Roo.EventObjectImpl.prototype;
59309      * e.keyCode = e.TAB
59310      *
59311      * 
59312      */
59313     onEditorKey : function(field, e){
59314         
59315         var k = e.getKey(),
59316             newCell,
59317             g = this.grid,
59318             ed = g.activeEditor,
59319             forward = false;
59320         ///Roo.log('onEditorKey' + k);
59321         
59322         
59323         if (this.enter_is_tab && k == e.ENTER) {
59324             k = e.TAB;
59325         }
59326         
59327         if(k == e.TAB){
59328             if(e.shiftKey){
59329                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59330             }else{
59331                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59332                 forward = true;
59333             }
59334             
59335             e.stopEvent();
59336             
59337         } else if(k == e.ENTER &&  !e.ctrlKey){
59338             ed.completeEdit();
59339             e.stopEvent();
59340             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59341         
59342                 } else if(k == e.ESC){
59343             ed.cancelEdit();
59344         }
59345                 
59346         if (newCell) {
59347             var ecall = { cell : newCell, forward : forward };
59348             this.fireEvent('beforeeditnext', ecall );
59349             newCell = ecall.cell;
59350                         forward = ecall.forward;
59351         }
59352                 
59353         if(newCell){
59354             //Roo.log('next cell after edit');
59355             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59356         } else if (forward) {
59357             // tabbed past last
59358             this.fireEvent.defer(100, this, ['tabend',this]);
59359         }
59360     }
59361 });/*
59362  * Based on:
59363  * Ext JS Library 1.1.1
59364  * Copyright(c) 2006-2007, Ext JS, LLC.
59365  *
59366  * Originally Released Under LGPL - original licence link has changed is not relivant.
59367  *
59368  * Fork - LGPL
59369  * <script type="text/javascript">
59370  */
59371  
59372 /**
59373  * @class Roo.grid.EditorGrid
59374  * @extends Roo.grid.Grid
59375  * Class for creating and editable grid.
59376  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59377  * The container MUST have some type of size defined for the grid to fill. The container will be 
59378  * automatically set to position relative if it isn't already.
59379  * @param {Object} dataSource The data model to bind to
59380  * @param {Object} colModel The column model with info about this grid's columns
59381  */
59382 Roo.grid.EditorGrid = function(container, config){
59383     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59384     this.getGridEl().addClass("xedit-grid");
59385
59386     if(!this.selModel){
59387         this.selModel = new Roo.grid.CellSelectionModel();
59388     }
59389
59390     this.activeEditor = null;
59391
59392         this.addEvents({
59393             /**
59394              * @event beforeedit
59395              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59396              * <ul style="padding:5px;padding-left:16px;">
59397              * <li>grid - This grid</li>
59398              * <li>record - The record being edited</li>
59399              * <li>field - The field name being edited</li>
59400              * <li>value - The value for the field being edited.</li>
59401              * <li>row - The grid row index</li>
59402              * <li>column - The grid column index</li>
59403              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59404              * </ul>
59405              * @param {Object} e An edit event (see above for description)
59406              */
59407             "beforeedit" : true,
59408             /**
59409              * @event afteredit
59410              * Fires after a cell is edited. <br />
59411              * <ul style="padding:5px;padding-left:16px;">
59412              * <li>grid - This grid</li>
59413              * <li>record - The record being edited</li>
59414              * <li>field - The field name being edited</li>
59415              * <li>value - The value being set</li>
59416              * <li>originalValue - The original value for the field, before the edit.</li>
59417              * <li>row - The grid row index</li>
59418              * <li>column - The grid column index</li>
59419              * </ul>
59420              * @param {Object} e An edit event (see above for description)
59421              */
59422             "afteredit" : true,
59423             /**
59424              * @event validateedit
59425              * Fires after a cell is edited, but before the value is set in the record. 
59426          * You can use this to modify the value being set in the field, Return false
59427              * to cancel the change. The edit event object has the following properties <br />
59428              * <ul style="padding:5px;padding-left:16px;">
59429          * <li>editor - This editor</li>
59430              * <li>grid - This grid</li>
59431              * <li>record - The record being edited</li>
59432              * <li>field - The field name being edited</li>
59433              * <li>value - The value being set</li>
59434              * <li>originalValue - The original value for the field, before the edit.</li>
59435              * <li>row - The grid row index</li>
59436              * <li>column - The grid column index</li>
59437              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59438              * </ul>
59439              * @param {Object} e An edit event (see above for description)
59440              */
59441             "validateedit" : true
59442         });
59443     this.on("bodyscroll", this.stopEditing,  this);
59444     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59445 };
59446
59447 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59448     /**
59449      * @cfg {Number} clicksToEdit
59450      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59451      */
59452     clicksToEdit: 2,
59453
59454     // private
59455     isEditor : true,
59456     // private
59457     trackMouseOver: false, // causes very odd FF errors
59458
59459     onCellDblClick : function(g, row, col){
59460         this.startEditing(row, col);
59461     },
59462
59463     onEditComplete : function(ed, value, startValue){
59464         this.editing = false;
59465         this.activeEditor = null;
59466         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59467         var r = ed.record;
59468         var field = this.colModel.getDataIndex(ed.col);
59469         var e = {
59470             grid: this,
59471             record: r,
59472             field: field,
59473             originalValue: startValue,
59474             value: value,
59475             row: ed.row,
59476             column: ed.col,
59477             cancel:false,
59478             editor: ed
59479         };
59480         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59481         cell.show();
59482           
59483         if(String(value) !== String(startValue)){
59484             
59485             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59486                 r.set(field, e.value);
59487                 // if we are dealing with a combo box..
59488                 // then we also set the 'name' colum to be the displayField
59489                 if (ed.field.displayField && ed.field.name) {
59490                     r.set(ed.field.name, ed.field.el.dom.value);
59491                 }
59492                 
59493                 delete e.cancel; //?? why!!!
59494                 this.fireEvent("afteredit", e);
59495             }
59496         } else {
59497             this.fireEvent("afteredit", e); // always fire it!
59498         }
59499         this.view.focusCell(ed.row, ed.col);
59500     },
59501
59502     /**
59503      * Starts editing the specified for the specified row/column
59504      * @param {Number} rowIndex
59505      * @param {Number} colIndex
59506      */
59507     startEditing : function(row, col){
59508         this.stopEditing();
59509         if(this.colModel.isCellEditable(col, row)){
59510             this.view.ensureVisible(row, col, true);
59511           
59512             var r = this.dataSource.getAt(row);
59513             var field = this.colModel.getDataIndex(col);
59514             var cell = Roo.get(this.view.getCell(row,col));
59515             var e = {
59516                 grid: this,
59517                 record: r,
59518                 field: field,
59519                 value: r.data[field],
59520                 row: row,
59521                 column: col,
59522                 cancel:false 
59523             };
59524             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59525                 this.editing = true;
59526                 var ed = this.colModel.getCellEditor(col, row);
59527                 
59528                 if (!ed) {
59529                     return;
59530                 }
59531                 if(!ed.rendered){
59532                     ed.render(ed.parentEl || document.body);
59533                 }
59534                 ed.field.reset();
59535                
59536                 cell.hide();
59537                 
59538                 (function(){ // complex but required for focus issues in safari, ie and opera
59539                     ed.row = row;
59540                     ed.col = col;
59541                     ed.record = r;
59542                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59543                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59544                     this.activeEditor = ed;
59545                     var v = r.data[field];
59546                     ed.startEdit(this.view.getCell(row, col), v);
59547                     // combo's with 'displayField and name set
59548                     if (ed.field.displayField && ed.field.name) {
59549                         ed.field.el.dom.value = r.data[ed.field.name];
59550                     }
59551                     
59552                     
59553                 }).defer(50, this);
59554             }
59555         }
59556     },
59557         
59558     /**
59559      * Stops any active editing
59560      */
59561     stopEditing : function(){
59562         if(this.activeEditor){
59563             this.activeEditor.completeEdit();
59564         }
59565         this.activeEditor = null;
59566     },
59567         
59568          /**
59569      * Called to get grid's drag proxy text, by default returns this.ddText.
59570      * @return {String}
59571      */
59572     getDragDropText : function(){
59573         var count = this.selModel.getSelectedCell() ? 1 : 0;
59574         return String.format(this.ddText, count, count == 1 ? '' : 's');
59575     }
59576         
59577 });/*
59578  * Based on:
59579  * Ext JS Library 1.1.1
59580  * Copyright(c) 2006-2007, Ext JS, LLC.
59581  *
59582  * Originally Released Under LGPL - original licence link has changed is not relivant.
59583  *
59584  * Fork - LGPL
59585  * <script type="text/javascript">
59586  */
59587
59588 // private - not really -- you end up using it !
59589 // This is a support class used internally by the Grid components
59590
59591 /**
59592  * @class Roo.grid.GridEditor
59593  * @extends Roo.Editor
59594  * Class for creating and editable grid elements.
59595  * @param {Object} config any settings (must include field)
59596  */
59597 Roo.grid.GridEditor = function(field, config){
59598     if (!config && field.field) {
59599         config = field;
59600         field = Roo.factory(config.field, Roo.form);
59601     }
59602     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59603     field.monitorTab = false;
59604 };
59605
59606 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59607     
59608     /**
59609      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59610      */
59611     
59612     alignment: "tl-tl",
59613     autoSize: "width",
59614     hideEl : false,
59615     cls: "x-small-editor x-grid-editor",
59616     shim:false,
59617     shadow:"frame"
59618 });/*
59619  * Based on:
59620  * Ext JS Library 1.1.1
59621  * Copyright(c) 2006-2007, Ext JS, LLC.
59622  *
59623  * Originally Released Under LGPL - original licence link has changed is not relivant.
59624  *
59625  * Fork - LGPL
59626  * <script type="text/javascript">
59627  */
59628   
59629
59630   
59631 Roo.grid.PropertyRecord = Roo.data.Record.create([
59632     {name:'name',type:'string'},  'value'
59633 ]);
59634
59635
59636 Roo.grid.PropertyStore = function(grid, source){
59637     this.grid = grid;
59638     this.store = new Roo.data.Store({
59639         recordType : Roo.grid.PropertyRecord
59640     });
59641     this.store.on('update', this.onUpdate,  this);
59642     if(source){
59643         this.setSource(source);
59644     }
59645     Roo.grid.PropertyStore.superclass.constructor.call(this);
59646 };
59647
59648
59649
59650 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59651     setSource : function(o){
59652         this.source = o;
59653         this.store.removeAll();
59654         var data = [];
59655         for(var k in o){
59656             if(this.isEditableValue(o[k])){
59657                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59658             }
59659         }
59660         this.store.loadRecords({records: data}, {}, true);
59661     },
59662
59663     onUpdate : function(ds, record, type){
59664         if(type == Roo.data.Record.EDIT){
59665             var v = record.data['value'];
59666             var oldValue = record.modified['value'];
59667             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59668                 this.source[record.id] = v;
59669                 record.commit();
59670                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59671             }else{
59672                 record.reject();
59673             }
59674         }
59675     },
59676
59677     getProperty : function(row){
59678        return this.store.getAt(row);
59679     },
59680
59681     isEditableValue: function(val){
59682         if(val && val instanceof Date){
59683             return true;
59684         }else if(typeof val == 'object' || typeof val == 'function'){
59685             return false;
59686         }
59687         return true;
59688     },
59689
59690     setValue : function(prop, value){
59691         this.source[prop] = value;
59692         this.store.getById(prop).set('value', value);
59693     },
59694
59695     getSource : function(){
59696         return this.source;
59697     }
59698 });
59699
59700 Roo.grid.PropertyColumnModel = function(grid, store){
59701     this.grid = grid;
59702     var g = Roo.grid;
59703     g.PropertyColumnModel.superclass.constructor.call(this, [
59704         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59705         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59706     ]);
59707     this.store = store;
59708     this.bselect = Roo.DomHelper.append(document.body, {
59709         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59710             {tag: 'option', value: 'true', html: 'true'},
59711             {tag: 'option', value: 'false', html: 'false'}
59712         ]
59713     });
59714     Roo.id(this.bselect);
59715     var f = Roo.form;
59716     this.editors = {
59717         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59718         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59719         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59720         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59721         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59722     };
59723     this.renderCellDelegate = this.renderCell.createDelegate(this);
59724     this.renderPropDelegate = this.renderProp.createDelegate(this);
59725 };
59726
59727 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59728     
59729     
59730     nameText : 'Name',
59731     valueText : 'Value',
59732     
59733     dateFormat : 'm/j/Y',
59734     
59735     
59736     renderDate : function(dateVal){
59737         return dateVal.dateFormat(this.dateFormat);
59738     },
59739
59740     renderBool : function(bVal){
59741         return bVal ? 'true' : 'false';
59742     },
59743
59744     isCellEditable : function(colIndex, rowIndex){
59745         return colIndex == 1;
59746     },
59747
59748     getRenderer : function(col){
59749         return col == 1 ?
59750             this.renderCellDelegate : this.renderPropDelegate;
59751     },
59752
59753     renderProp : function(v){
59754         return this.getPropertyName(v);
59755     },
59756
59757     renderCell : function(val){
59758         var rv = val;
59759         if(val instanceof Date){
59760             rv = this.renderDate(val);
59761         }else if(typeof val == 'boolean'){
59762             rv = this.renderBool(val);
59763         }
59764         return Roo.util.Format.htmlEncode(rv);
59765     },
59766
59767     getPropertyName : function(name){
59768         var pn = this.grid.propertyNames;
59769         return pn && pn[name] ? pn[name] : name;
59770     },
59771
59772     getCellEditor : function(colIndex, rowIndex){
59773         var p = this.store.getProperty(rowIndex);
59774         var n = p.data['name'], val = p.data['value'];
59775         
59776         if(typeof(this.grid.customEditors[n]) == 'string'){
59777             return this.editors[this.grid.customEditors[n]];
59778         }
59779         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59780             return this.grid.customEditors[n];
59781         }
59782         if(val instanceof Date){
59783             return this.editors['date'];
59784         }else if(typeof val == 'number'){
59785             return this.editors['number'];
59786         }else if(typeof val == 'boolean'){
59787             return this.editors['boolean'];
59788         }else{
59789             return this.editors['string'];
59790         }
59791     }
59792 });
59793
59794 /**
59795  * @class Roo.grid.PropertyGrid
59796  * @extends Roo.grid.EditorGrid
59797  * This class represents the  interface of a component based property grid control.
59798  * <br><br>Usage:<pre><code>
59799  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59800       
59801  });
59802  // set any options
59803  grid.render();
59804  * </code></pre>
59805   
59806  * @constructor
59807  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59808  * The container MUST have some type of size defined for the grid to fill. The container will be
59809  * automatically set to position relative if it isn't already.
59810  * @param {Object} config A config object that sets properties on this grid.
59811  */
59812 Roo.grid.PropertyGrid = function(container, config){
59813     config = config || {};
59814     var store = new Roo.grid.PropertyStore(this);
59815     this.store = store;
59816     var cm = new Roo.grid.PropertyColumnModel(this, store);
59817     store.store.sort('name', 'ASC');
59818     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59819         ds: store.store,
59820         cm: cm,
59821         enableColLock:false,
59822         enableColumnMove:false,
59823         stripeRows:false,
59824         trackMouseOver: false,
59825         clicksToEdit:1
59826     }, config));
59827     this.getGridEl().addClass('x-props-grid');
59828     this.lastEditRow = null;
59829     this.on('columnresize', this.onColumnResize, this);
59830     this.addEvents({
59831          /**
59832              * @event beforepropertychange
59833              * Fires before a property changes (return false to stop?)
59834              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59835              * @param {String} id Record Id
59836              * @param {String} newval New Value
59837          * @param {String} oldval Old Value
59838              */
59839         "beforepropertychange": true,
59840         /**
59841              * @event propertychange
59842              * Fires after a property changes
59843              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59844              * @param {String} id Record Id
59845              * @param {String} newval New Value
59846          * @param {String} oldval Old Value
59847              */
59848         "propertychange": true
59849     });
59850     this.customEditors = this.customEditors || {};
59851 };
59852 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59853     
59854      /**
59855      * @cfg {Object} customEditors map of colnames=> custom editors.
59856      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59857      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59858      * false disables editing of the field.
59859          */
59860     
59861       /**
59862      * @cfg {Object} propertyNames map of property Names to their displayed value
59863          */
59864     
59865     render : function(){
59866         Roo.grid.PropertyGrid.superclass.render.call(this);
59867         this.autoSize.defer(100, this);
59868     },
59869
59870     autoSize : function(){
59871         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59872         if(this.view){
59873             this.view.fitColumns();
59874         }
59875     },
59876
59877     onColumnResize : function(){
59878         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59879         this.autoSize();
59880     },
59881     /**
59882      * Sets the data for the Grid
59883      * accepts a Key => Value object of all the elements avaiable.
59884      * @param {Object} data  to appear in grid.
59885      */
59886     setSource : function(source){
59887         this.store.setSource(source);
59888         //this.autoSize();
59889     },
59890     /**
59891      * Gets all the data from the grid.
59892      * @return {Object} data  data stored in grid
59893      */
59894     getSource : function(){
59895         return this.store.getSource();
59896     }
59897 });/*
59898   
59899  * Licence LGPL
59900  
59901  */
59902  
59903 /**
59904  * @class Roo.grid.Calendar
59905  * @extends Roo.util.Grid
59906  * This class extends the Grid to provide a calendar widget
59907  * <br><br>Usage:<pre><code>
59908  var grid = new Roo.grid.Calendar("my-container-id", {
59909      ds: myDataStore,
59910      cm: myColModel,
59911      selModel: mySelectionModel,
59912      autoSizeColumns: true,
59913      monitorWindowResize: false,
59914      trackMouseOver: true
59915      eventstore : real data store..
59916  });
59917  // set any options
59918  grid.render();
59919   
59920   * @constructor
59921  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59922  * The container MUST have some type of size defined for the grid to fill. The container will be
59923  * automatically set to position relative if it isn't already.
59924  * @param {Object} config A config object that sets properties on this grid.
59925  */
59926 Roo.grid.Calendar = function(container, config){
59927         // initialize the container
59928         this.container = Roo.get(container);
59929         this.container.update("");
59930         this.container.setStyle("overflow", "hidden");
59931     this.container.addClass('x-grid-container');
59932
59933     this.id = this.container.id;
59934
59935     Roo.apply(this, config);
59936     // check and correct shorthanded configs
59937     
59938     var rows = [];
59939     var d =1;
59940     for (var r = 0;r < 6;r++) {
59941         
59942         rows[r]=[];
59943         for (var c =0;c < 7;c++) {
59944             rows[r][c]= '';
59945         }
59946     }
59947     if (this.eventStore) {
59948         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59949         this.eventStore.on('load',this.onLoad, this);
59950         this.eventStore.on('beforeload',this.clearEvents, this);
59951          
59952     }
59953     
59954     this.dataSource = new Roo.data.Store({
59955             proxy: new Roo.data.MemoryProxy(rows),
59956             reader: new Roo.data.ArrayReader({}, [
59957                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59958     });
59959
59960     this.dataSource.load();
59961     this.ds = this.dataSource;
59962     this.ds.xmodule = this.xmodule || false;
59963     
59964     
59965     var cellRender = function(v,x,r)
59966     {
59967         return String.format(
59968             '<div class="fc-day  fc-widget-content"><div>' +
59969                 '<div class="fc-event-container"></div>' +
59970                 '<div class="fc-day-number">{0}</div>'+
59971                 
59972                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59973             '</div></div>', v);
59974     
59975     }
59976     
59977     
59978     this.colModel = new Roo.grid.ColumnModel( [
59979         {
59980             xtype: 'ColumnModel',
59981             xns: Roo.grid,
59982             dataIndex : 'weekday0',
59983             header : 'Sunday',
59984             renderer : cellRender
59985         },
59986         {
59987             xtype: 'ColumnModel',
59988             xns: Roo.grid,
59989             dataIndex : 'weekday1',
59990             header : 'Monday',
59991             renderer : cellRender
59992         },
59993         {
59994             xtype: 'ColumnModel',
59995             xns: Roo.grid,
59996             dataIndex : 'weekday2',
59997             header : 'Tuesday',
59998             renderer : cellRender
59999         },
60000         {
60001             xtype: 'ColumnModel',
60002             xns: Roo.grid,
60003             dataIndex : 'weekday3',
60004             header : 'Wednesday',
60005             renderer : cellRender
60006         },
60007         {
60008             xtype: 'ColumnModel',
60009             xns: Roo.grid,
60010             dataIndex : 'weekday4',
60011             header : 'Thursday',
60012             renderer : cellRender
60013         },
60014         {
60015             xtype: 'ColumnModel',
60016             xns: Roo.grid,
60017             dataIndex : 'weekday5',
60018             header : 'Friday',
60019             renderer : cellRender
60020         },
60021         {
60022             xtype: 'ColumnModel',
60023             xns: Roo.grid,
60024             dataIndex : 'weekday6',
60025             header : 'Saturday',
60026             renderer : cellRender
60027         }
60028     ]);
60029     this.cm = this.colModel;
60030     this.cm.xmodule = this.xmodule || false;
60031  
60032         
60033           
60034     //this.selModel = new Roo.grid.CellSelectionModel();
60035     //this.sm = this.selModel;
60036     //this.selModel.init(this);
60037     
60038     
60039     if(this.width){
60040         this.container.setWidth(this.width);
60041     }
60042
60043     if(this.height){
60044         this.container.setHeight(this.height);
60045     }
60046     /** @private */
60047         this.addEvents({
60048         // raw events
60049         /**
60050          * @event click
60051          * The raw click event for the entire grid.
60052          * @param {Roo.EventObject} e
60053          */
60054         "click" : true,
60055         /**
60056          * @event dblclick
60057          * The raw dblclick event for the entire grid.
60058          * @param {Roo.EventObject} e
60059          */
60060         "dblclick" : true,
60061         /**
60062          * @event contextmenu
60063          * The raw contextmenu event for the entire grid.
60064          * @param {Roo.EventObject} e
60065          */
60066         "contextmenu" : true,
60067         /**
60068          * @event mousedown
60069          * The raw mousedown event for the entire grid.
60070          * @param {Roo.EventObject} e
60071          */
60072         "mousedown" : true,
60073         /**
60074          * @event mouseup
60075          * The raw mouseup event for the entire grid.
60076          * @param {Roo.EventObject} e
60077          */
60078         "mouseup" : true,
60079         /**
60080          * @event mouseover
60081          * The raw mouseover event for the entire grid.
60082          * @param {Roo.EventObject} e
60083          */
60084         "mouseover" : true,
60085         /**
60086          * @event mouseout
60087          * The raw mouseout event for the entire grid.
60088          * @param {Roo.EventObject} e
60089          */
60090         "mouseout" : true,
60091         /**
60092          * @event keypress
60093          * The raw keypress event for the entire grid.
60094          * @param {Roo.EventObject} e
60095          */
60096         "keypress" : true,
60097         /**
60098          * @event keydown
60099          * The raw keydown event for the entire grid.
60100          * @param {Roo.EventObject} e
60101          */
60102         "keydown" : true,
60103
60104         // custom events
60105
60106         /**
60107          * @event cellclick
60108          * Fires when a cell is clicked
60109          * @param {Grid} this
60110          * @param {Number} rowIndex
60111          * @param {Number} columnIndex
60112          * @param {Roo.EventObject} e
60113          */
60114         "cellclick" : true,
60115         /**
60116          * @event celldblclick
60117          * Fires when a cell is double clicked
60118          * @param {Grid} this
60119          * @param {Number} rowIndex
60120          * @param {Number} columnIndex
60121          * @param {Roo.EventObject} e
60122          */
60123         "celldblclick" : true,
60124         /**
60125          * @event rowclick
60126          * Fires when a row is clicked
60127          * @param {Grid} this
60128          * @param {Number} rowIndex
60129          * @param {Roo.EventObject} e
60130          */
60131         "rowclick" : true,
60132         /**
60133          * @event rowdblclick
60134          * Fires when a row is double clicked
60135          * @param {Grid} this
60136          * @param {Number} rowIndex
60137          * @param {Roo.EventObject} e
60138          */
60139         "rowdblclick" : true,
60140         /**
60141          * @event headerclick
60142          * Fires when a header is clicked
60143          * @param {Grid} this
60144          * @param {Number} columnIndex
60145          * @param {Roo.EventObject} e
60146          */
60147         "headerclick" : true,
60148         /**
60149          * @event headerdblclick
60150          * Fires when a header cell is double clicked
60151          * @param {Grid} this
60152          * @param {Number} columnIndex
60153          * @param {Roo.EventObject} e
60154          */
60155         "headerdblclick" : true,
60156         /**
60157          * @event rowcontextmenu
60158          * Fires when a row is right clicked
60159          * @param {Grid} this
60160          * @param {Number} rowIndex
60161          * @param {Roo.EventObject} e
60162          */
60163         "rowcontextmenu" : true,
60164         /**
60165          * @event cellcontextmenu
60166          * Fires when a cell is right clicked
60167          * @param {Grid} this
60168          * @param {Number} rowIndex
60169          * @param {Number} cellIndex
60170          * @param {Roo.EventObject} e
60171          */
60172          "cellcontextmenu" : true,
60173         /**
60174          * @event headercontextmenu
60175          * Fires when a header is right clicked
60176          * @param {Grid} this
60177          * @param {Number} columnIndex
60178          * @param {Roo.EventObject} e
60179          */
60180         "headercontextmenu" : true,
60181         /**
60182          * @event bodyscroll
60183          * Fires when the body element is scrolled
60184          * @param {Number} scrollLeft
60185          * @param {Number} scrollTop
60186          */
60187         "bodyscroll" : true,
60188         /**
60189          * @event columnresize
60190          * Fires when the user resizes a column
60191          * @param {Number} columnIndex
60192          * @param {Number} newSize
60193          */
60194         "columnresize" : true,
60195         /**
60196          * @event columnmove
60197          * Fires when the user moves a column
60198          * @param {Number} oldIndex
60199          * @param {Number} newIndex
60200          */
60201         "columnmove" : true,
60202         /**
60203          * @event startdrag
60204          * Fires when row(s) start being dragged
60205          * @param {Grid} this
60206          * @param {Roo.GridDD} dd The drag drop object
60207          * @param {event} e The raw browser event
60208          */
60209         "startdrag" : true,
60210         /**
60211          * @event enddrag
60212          * Fires when a drag operation is complete
60213          * @param {Grid} this
60214          * @param {Roo.GridDD} dd The drag drop object
60215          * @param {event} e The raw browser event
60216          */
60217         "enddrag" : true,
60218         /**
60219          * @event dragdrop
60220          * Fires when dragged row(s) are dropped on a valid DD target
60221          * @param {Grid} this
60222          * @param {Roo.GridDD} dd The drag drop object
60223          * @param {String} targetId The target drag drop object
60224          * @param {event} e The raw browser event
60225          */
60226         "dragdrop" : true,
60227         /**
60228          * @event dragover
60229          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60230          * @param {Grid} this
60231          * @param {Roo.GridDD} dd The drag drop object
60232          * @param {String} targetId The target drag drop object
60233          * @param {event} e The raw browser event
60234          */
60235         "dragover" : true,
60236         /**
60237          * @event dragenter
60238          *  Fires when the dragged row(s) first cross another DD target while being dragged
60239          * @param {Grid} this
60240          * @param {Roo.GridDD} dd The drag drop object
60241          * @param {String} targetId The target drag drop object
60242          * @param {event} e The raw browser event
60243          */
60244         "dragenter" : true,
60245         /**
60246          * @event dragout
60247          * Fires when the dragged row(s) leave another DD target while being dragged
60248          * @param {Grid} this
60249          * @param {Roo.GridDD} dd The drag drop object
60250          * @param {String} targetId The target drag drop object
60251          * @param {event} e The raw browser event
60252          */
60253         "dragout" : true,
60254         /**
60255          * @event rowclass
60256          * Fires when a row is rendered, so you can change add a style to it.
60257          * @param {GridView} gridview   The grid view
60258          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60259          */
60260         'rowclass' : true,
60261
60262         /**
60263          * @event render
60264          * Fires when the grid is rendered
60265          * @param {Grid} grid
60266          */
60267         'render' : true,
60268             /**
60269              * @event select
60270              * Fires when a date is selected
60271              * @param {DatePicker} this
60272              * @param {Date} date The selected date
60273              */
60274         'select': true,
60275         /**
60276              * @event monthchange
60277              * Fires when the displayed month changes 
60278              * @param {DatePicker} this
60279              * @param {Date} date The selected month
60280              */
60281         'monthchange': true,
60282         /**
60283              * @event evententer
60284              * Fires when mouse over an event
60285              * @param {Calendar} this
60286              * @param {event} Event
60287              */
60288         'evententer': true,
60289         /**
60290              * @event eventleave
60291              * Fires when the mouse leaves an
60292              * @param {Calendar} this
60293              * @param {event}
60294              */
60295         'eventleave': true,
60296         /**
60297              * @event eventclick
60298              * Fires when the mouse click an
60299              * @param {Calendar} this
60300              * @param {event}
60301              */
60302         'eventclick': true,
60303         /**
60304              * @event eventrender
60305              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60306              * @param {Calendar} this
60307              * @param {data} data to be modified
60308              */
60309         'eventrender': true
60310         
60311     });
60312
60313     Roo.grid.Grid.superclass.constructor.call(this);
60314     this.on('render', function() {
60315         this.view.el.addClass('x-grid-cal'); 
60316         
60317         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60318
60319     },this);
60320     
60321     if (!Roo.grid.Calendar.style) {
60322         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60323             
60324             
60325             '.x-grid-cal .x-grid-col' :  {
60326                 height: 'auto !important',
60327                 'vertical-align': 'top'
60328             },
60329             '.x-grid-cal  .fc-event-hori' : {
60330                 height: '14px'
60331             }
60332              
60333             
60334         }, Roo.id());
60335     }
60336
60337     
60338     
60339 };
60340 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60341     /**
60342      * @cfg {Store} eventStore The store that loads events.
60343      */
60344     eventStore : 25,
60345
60346      
60347     activeDate : false,
60348     startDay : 0,
60349     autoWidth : true,
60350     monitorWindowResize : false,
60351
60352     
60353     resizeColumns : function() {
60354         var col = (this.view.el.getWidth() / 7) - 3;
60355         // loop through cols, and setWidth
60356         for(var i =0 ; i < 7 ; i++){
60357             this.cm.setColumnWidth(i, col);
60358         }
60359     },
60360      setDate :function(date) {
60361         
60362         Roo.log('setDate?');
60363         
60364         this.resizeColumns();
60365         var vd = this.activeDate;
60366         this.activeDate = date;
60367 //        if(vd && this.el){
60368 //            var t = date.getTime();
60369 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60370 //                Roo.log('using add remove');
60371 //                
60372 //                this.fireEvent('monthchange', this, date);
60373 //                
60374 //                this.cells.removeClass("fc-state-highlight");
60375 //                this.cells.each(function(c){
60376 //                   if(c.dateValue == t){
60377 //                       c.addClass("fc-state-highlight");
60378 //                       setTimeout(function(){
60379 //                            try{c.dom.firstChild.focus();}catch(e){}
60380 //                       }, 50);
60381 //                       return false;
60382 //                   }
60383 //                   return true;
60384 //                });
60385 //                return;
60386 //            }
60387 //        }
60388         
60389         var days = date.getDaysInMonth();
60390         
60391         var firstOfMonth = date.getFirstDateOfMonth();
60392         var startingPos = firstOfMonth.getDay()-this.startDay;
60393         
60394         if(startingPos < this.startDay){
60395             startingPos += 7;
60396         }
60397         
60398         var pm = date.add(Date.MONTH, -1);
60399         var prevStart = pm.getDaysInMonth()-startingPos;
60400 //        
60401         
60402         
60403         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60404         
60405         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60406         //this.cells.addClassOnOver('fc-state-hover');
60407         
60408         var cells = this.cells.elements;
60409         var textEls = this.textNodes;
60410         
60411         //Roo.each(cells, function(cell){
60412         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60413         //});
60414         
60415         days += startingPos;
60416
60417         // convert everything to numbers so it's fast
60418         var day = 86400000;
60419         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60420         //Roo.log(d);
60421         //Roo.log(pm);
60422         //Roo.log(prevStart);
60423         
60424         var today = new Date().clearTime().getTime();
60425         var sel = date.clearTime().getTime();
60426         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60427         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60428         var ddMatch = this.disabledDatesRE;
60429         var ddText = this.disabledDatesText;
60430         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60431         var ddaysText = this.disabledDaysText;
60432         var format = this.format;
60433         
60434         var setCellClass = function(cal, cell){
60435             
60436             //Roo.log('set Cell Class');
60437             cell.title = "";
60438             var t = d.getTime();
60439             
60440             //Roo.log(d);
60441             
60442             
60443             cell.dateValue = t;
60444             if(t == today){
60445                 cell.className += " fc-today";
60446                 cell.className += " fc-state-highlight";
60447                 cell.title = cal.todayText;
60448             }
60449             if(t == sel){
60450                 // disable highlight in other month..
60451                 cell.className += " fc-state-highlight";
60452                 
60453             }
60454             // disabling
60455             if(t < min) {
60456                 //cell.className = " fc-state-disabled";
60457                 cell.title = cal.minText;
60458                 return;
60459             }
60460             if(t > max) {
60461                 //cell.className = " fc-state-disabled";
60462                 cell.title = cal.maxText;
60463                 return;
60464             }
60465             if(ddays){
60466                 if(ddays.indexOf(d.getDay()) != -1){
60467                     // cell.title = ddaysText;
60468                    // cell.className = " fc-state-disabled";
60469                 }
60470             }
60471             if(ddMatch && format){
60472                 var fvalue = d.dateFormat(format);
60473                 if(ddMatch.test(fvalue)){
60474                     cell.title = ddText.replace("%0", fvalue);
60475                    cell.className = " fc-state-disabled";
60476                 }
60477             }
60478             
60479             if (!cell.initialClassName) {
60480                 cell.initialClassName = cell.dom.className;
60481             }
60482             
60483             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60484         };
60485
60486         var i = 0;
60487         
60488         for(; i < startingPos; i++) {
60489             cells[i].dayName =  (++prevStart);
60490             Roo.log(textEls[i]);
60491             d.setDate(d.getDate()+1);
60492             
60493             //cells[i].className = "fc-past fc-other-month";
60494             setCellClass(this, cells[i]);
60495         }
60496         
60497         var intDay = 0;
60498         
60499         for(; i < days; i++){
60500             intDay = i - startingPos + 1;
60501             cells[i].dayName =  (intDay);
60502             d.setDate(d.getDate()+1);
60503             
60504             cells[i].className = ''; // "x-date-active";
60505             setCellClass(this, cells[i]);
60506         }
60507         var extraDays = 0;
60508         
60509         for(; i < 42; i++) {
60510             //textEls[i].innerHTML = (++extraDays);
60511             
60512             d.setDate(d.getDate()+1);
60513             cells[i].dayName = (++extraDays);
60514             cells[i].className = "fc-future fc-other-month";
60515             setCellClass(this, cells[i]);
60516         }
60517         
60518         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60519         
60520         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60521         
60522         // this will cause all the cells to mis
60523         var rows= [];
60524         var i =0;
60525         for (var r = 0;r < 6;r++) {
60526             for (var c =0;c < 7;c++) {
60527                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60528             }    
60529         }
60530         
60531         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60532         for(i=0;i<cells.length;i++) {
60533             
60534             this.cells.elements[i].dayName = cells[i].dayName ;
60535             this.cells.elements[i].className = cells[i].className;
60536             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60537             this.cells.elements[i].title = cells[i].title ;
60538             this.cells.elements[i].dateValue = cells[i].dateValue ;
60539         }
60540         
60541         
60542         
60543         
60544         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60545         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60546         
60547         ////if(totalRows != 6){
60548             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60549            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60550        // }
60551         
60552         this.fireEvent('monthchange', this, date);
60553         
60554         
60555     },
60556  /**
60557      * Returns the grid's SelectionModel.
60558      * @return {SelectionModel}
60559      */
60560     getSelectionModel : function(){
60561         if(!this.selModel){
60562             this.selModel = new Roo.grid.CellSelectionModel();
60563         }
60564         return this.selModel;
60565     },
60566
60567     load: function() {
60568         this.eventStore.load()
60569         
60570         
60571         
60572     },
60573     
60574     findCell : function(dt) {
60575         dt = dt.clearTime().getTime();
60576         var ret = false;
60577         this.cells.each(function(c){
60578             //Roo.log("check " +c.dateValue + '?=' + dt);
60579             if(c.dateValue == dt){
60580                 ret = c;
60581                 return false;
60582             }
60583             return true;
60584         });
60585         
60586         return ret;
60587     },
60588     
60589     findCells : function(rec) {
60590         var s = rec.data.start_dt.clone().clearTime().getTime();
60591        // Roo.log(s);
60592         var e= rec.data.end_dt.clone().clearTime().getTime();
60593        // Roo.log(e);
60594         var ret = [];
60595         this.cells.each(function(c){
60596              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60597             
60598             if(c.dateValue > e){
60599                 return ;
60600             }
60601             if(c.dateValue < s){
60602                 return ;
60603             }
60604             ret.push(c);
60605         });
60606         
60607         return ret;    
60608     },
60609     
60610     findBestRow: function(cells)
60611     {
60612         var ret = 0;
60613         
60614         for (var i =0 ; i < cells.length;i++) {
60615             ret  = Math.max(cells[i].rows || 0,ret);
60616         }
60617         return ret;
60618         
60619     },
60620     
60621     
60622     addItem : function(rec)
60623     {
60624         // look for vertical location slot in
60625         var cells = this.findCells(rec);
60626         
60627         rec.row = this.findBestRow(cells);
60628         
60629         // work out the location.
60630         
60631         var crow = false;
60632         var rows = [];
60633         for(var i =0; i < cells.length; i++) {
60634             if (!crow) {
60635                 crow = {
60636                     start : cells[i],
60637                     end :  cells[i]
60638                 };
60639                 continue;
60640             }
60641             if (crow.start.getY() == cells[i].getY()) {
60642                 // on same row.
60643                 crow.end = cells[i];
60644                 continue;
60645             }
60646             // different row.
60647             rows.push(crow);
60648             crow = {
60649                 start: cells[i],
60650                 end : cells[i]
60651             };
60652             
60653         }
60654         
60655         rows.push(crow);
60656         rec.els = [];
60657         rec.rows = rows;
60658         rec.cells = cells;
60659         for (var i = 0; i < cells.length;i++) {
60660             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60661             
60662         }
60663         
60664         
60665     },
60666     
60667     clearEvents: function() {
60668         
60669         if (!this.eventStore.getCount()) {
60670             return;
60671         }
60672         // reset number of rows in cells.
60673         Roo.each(this.cells.elements, function(c){
60674             c.rows = 0;
60675         });
60676         
60677         this.eventStore.each(function(e) {
60678             this.clearEvent(e);
60679         },this);
60680         
60681     },
60682     
60683     clearEvent : function(ev)
60684     {
60685         if (ev.els) {
60686             Roo.each(ev.els, function(el) {
60687                 el.un('mouseenter' ,this.onEventEnter, this);
60688                 el.un('mouseleave' ,this.onEventLeave, this);
60689                 el.remove();
60690             },this);
60691             ev.els = [];
60692         }
60693     },
60694     
60695     
60696     renderEvent : function(ev,ctr) {
60697         if (!ctr) {
60698              ctr = this.view.el.select('.fc-event-container',true).first();
60699         }
60700         
60701          
60702         this.clearEvent(ev);
60703             //code
60704        
60705         
60706         
60707         ev.els = [];
60708         var cells = ev.cells;
60709         var rows = ev.rows;
60710         this.fireEvent('eventrender', this, ev);
60711         
60712         for(var i =0; i < rows.length; i++) {
60713             
60714             cls = '';
60715             if (i == 0) {
60716                 cls += ' fc-event-start';
60717             }
60718             if ((i+1) == rows.length) {
60719                 cls += ' fc-event-end';
60720             }
60721             
60722             //Roo.log(ev.data);
60723             // how many rows should it span..
60724             var cg = this.eventTmpl.append(ctr,Roo.apply({
60725                 fccls : cls
60726                 
60727             }, ev.data) , true);
60728             
60729             
60730             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60731             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60732             cg.on('click', this.onEventClick, this, ev);
60733             
60734             ev.els.push(cg);
60735             
60736             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60737             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60738             //Roo.log(cg);
60739              
60740             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60741             cg.setWidth(ebox.right - sbox.x -2);
60742         }
60743     },
60744     
60745     renderEvents: function()
60746     {   
60747         // first make sure there is enough space..
60748         
60749         if (!this.eventTmpl) {
60750             this.eventTmpl = new Roo.Template(
60751                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60752                     '<div class="fc-event-inner">' +
60753                         '<span class="fc-event-time">{time}</span>' +
60754                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60755                     '</div>' +
60756                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60757                 '</div>'
60758             );
60759                 
60760         }
60761                
60762         
60763         
60764         this.cells.each(function(c) {
60765             //Roo.log(c.select('.fc-day-content div',true).first());
60766             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60767         });
60768         
60769         var ctr = this.view.el.select('.fc-event-container',true).first();
60770         
60771         var cls;
60772         this.eventStore.each(function(ev){
60773             
60774             this.renderEvent(ev);
60775              
60776              
60777         }, this);
60778         this.view.layout();
60779         
60780     },
60781     
60782     onEventEnter: function (e, el,event,d) {
60783         this.fireEvent('evententer', this, el, event);
60784     },
60785     
60786     onEventLeave: function (e, el,event,d) {
60787         this.fireEvent('eventleave', this, el, event);
60788     },
60789     
60790     onEventClick: function (e, el,event,d) {
60791         this.fireEvent('eventclick', this, el, event);
60792     },
60793     
60794     onMonthChange: function () {
60795         this.store.load();
60796     },
60797     
60798     onLoad: function () {
60799         
60800         //Roo.log('calendar onload');
60801 //         
60802         if(this.eventStore.getCount() > 0){
60803             
60804            
60805             
60806             this.eventStore.each(function(d){
60807                 
60808                 
60809                 // FIXME..
60810                 var add =   d.data;
60811                 if (typeof(add.end_dt) == 'undefined')  {
60812                     Roo.log("Missing End time in calendar data: ");
60813                     Roo.log(d);
60814                     return;
60815                 }
60816                 if (typeof(add.start_dt) == 'undefined')  {
60817                     Roo.log("Missing Start time in calendar data: ");
60818                     Roo.log(d);
60819                     return;
60820                 }
60821                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60822                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60823                 add.id = add.id || d.id;
60824                 add.title = add.title || '??';
60825                 
60826                 this.addItem(d);
60827                 
60828              
60829             },this);
60830         }
60831         
60832         this.renderEvents();
60833     }
60834     
60835
60836 });
60837 /*
60838  grid : {
60839                 xtype: 'Grid',
60840                 xns: Roo.grid,
60841                 listeners : {
60842                     render : function ()
60843                     {
60844                         _this.grid = this;
60845                         
60846                         if (!this.view.el.hasClass('course-timesheet')) {
60847                             this.view.el.addClass('course-timesheet');
60848                         }
60849                         if (this.tsStyle) {
60850                             this.ds.load({});
60851                             return; 
60852                         }
60853                         Roo.log('width');
60854                         Roo.log(_this.grid.view.el.getWidth());
60855                         
60856                         
60857                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60858                             '.course-timesheet .x-grid-row' : {
60859                                 height: '80px'
60860                             },
60861                             '.x-grid-row td' : {
60862                                 'vertical-align' : 0
60863                             },
60864                             '.course-edit-link' : {
60865                                 'color' : 'blue',
60866                                 'text-overflow' : 'ellipsis',
60867                                 'overflow' : 'hidden',
60868                                 'white-space' : 'nowrap',
60869                                 'cursor' : 'pointer'
60870                             },
60871                             '.sub-link' : {
60872                                 'color' : 'green'
60873                             },
60874                             '.de-act-sup-link' : {
60875                                 'color' : 'purple',
60876                                 'text-decoration' : 'line-through'
60877                             },
60878                             '.de-act-link' : {
60879                                 'color' : 'red',
60880                                 'text-decoration' : 'line-through'
60881                             },
60882                             '.course-timesheet .course-highlight' : {
60883                                 'border-top-style': 'dashed !important',
60884                                 'border-bottom-bottom': 'dashed !important'
60885                             },
60886                             '.course-timesheet .course-item' : {
60887                                 'font-family'   : 'tahoma, arial, helvetica',
60888                                 'font-size'     : '11px',
60889                                 'overflow'      : 'hidden',
60890                                 'padding-left'  : '10px',
60891                                 'padding-right' : '10px',
60892                                 'padding-top' : '10px' 
60893                             }
60894                             
60895                         }, Roo.id());
60896                                 this.ds.load({});
60897                     }
60898                 },
60899                 autoWidth : true,
60900                 monitorWindowResize : false,
60901                 cellrenderer : function(v,x,r)
60902                 {
60903                     return v;
60904                 },
60905                 sm : {
60906                     xtype: 'CellSelectionModel',
60907                     xns: Roo.grid
60908                 },
60909                 dataSource : {
60910                     xtype: 'Store',
60911                     xns: Roo.data,
60912                     listeners : {
60913                         beforeload : function (_self, options)
60914                         {
60915                             options.params = options.params || {};
60916                             options.params._month = _this.monthField.getValue();
60917                             options.params.limit = 9999;
60918                             options.params['sort'] = 'when_dt';    
60919                             options.params['dir'] = 'ASC';    
60920                             this.proxy.loadResponse = this.loadResponse;
60921                             Roo.log("load?");
60922                             //this.addColumns();
60923                         },
60924                         load : function (_self, records, options)
60925                         {
60926                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60927                                 // if you click on the translation.. you can edit it...
60928                                 var el = Roo.get(this);
60929                                 var id = el.dom.getAttribute('data-id');
60930                                 var d = el.dom.getAttribute('data-date');
60931                                 var t = el.dom.getAttribute('data-time');
60932                                 //var id = this.child('span').dom.textContent;
60933                                 
60934                                 //Roo.log(this);
60935                                 Pman.Dialog.CourseCalendar.show({
60936                                     id : id,
60937                                     when_d : d,
60938                                     when_t : t,
60939                                     productitem_active : id ? 1 : 0
60940                                 }, function() {
60941                                     _this.grid.ds.load({});
60942                                 });
60943                            
60944                            });
60945                            
60946                            _this.panel.fireEvent('resize', [ '', '' ]);
60947                         }
60948                     },
60949                     loadResponse : function(o, success, response){
60950                             // this is overridden on before load..
60951                             
60952                             Roo.log("our code?");       
60953                             //Roo.log(success);
60954                             //Roo.log(response)
60955                             delete this.activeRequest;
60956                             if(!success){
60957                                 this.fireEvent("loadexception", this, o, response);
60958                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60959                                 return;
60960                             }
60961                             var result;
60962                             try {
60963                                 result = o.reader.read(response);
60964                             }catch(e){
60965                                 Roo.log("load exception?");
60966                                 this.fireEvent("loadexception", this, o, response, e);
60967                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60968                                 return;
60969                             }
60970                             Roo.log("ready...");        
60971                             // loop through result.records;
60972                             // and set this.tdate[date] = [] << array of records..
60973                             _this.tdata  = {};
60974                             Roo.each(result.records, function(r){
60975                                 //Roo.log(r.data);
60976                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60977                                     _this.tdata[r.data.when_dt.format('j')] = [];
60978                                 }
60979                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60980                             });
60981                             
60982                             //Roo.log(_this.tdata);
60983                             
60984                             result.records = [];
60985                             result.totalRecords = 6;
60986                     
60987                             // let's generate some duumy records for the rows.
60988                             //var st = _this.dateField.getValue();
60989                             
60990                             // work out monday..
60991                             //st = st.add(Date.DAY, -1 * st.format('w'));
60992                             
60993                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60994                             
60995                             var firstOfMonth = date.getFirstDayOfMonth();
60996                             var days = date.getDaysInMonth();
60997                             var d = 1;
60998                             var firstAdded = false;
60999                             for (var i = 0; i < result.totalRecords ; i++) {
61000                                 //var d= st.add(Date.DAY, i);
61001                                 var row = {};
61002                                 var added = 0;
61003                                 for(var w = 0 ; w < 7 ; w++){
61004                                     if(!firstAdded && firstOfMonth != w){
61005                                         continue;
61006                                     }
61007                                     if(d > days){
61008                                         continue;
61009                                     }
61010                                     firstAdded = true;
61011                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
61012                                     row['weekday'+w] = String.format(
61013                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
61014                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
61015                                                     d,
61016                                                     date.format('Y-m-')+dd
61017                                                 );
61018                                     added++;
61019                                     if(typeof(_this.tdata[d]) != 'undefined'){
61020                                         Roo.each(_this.tdata[d], function(r){
61021                                             var is_sub = '';
61022                                             var deactive = '';
61023                                             var id = r.id;
61024                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
61025                                             if(r.parent_id*1>0){
61026                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
61027                                                 id = r.parent_id;
61028                                             }
61029                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
61030                                                 deactive = 'de-act-link';
61031                                             }
61032                                             
61033                                             row['weekday'+w] += String.format(
61034                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
61035                                                     id, //0
61036                                                     r.product_id_name, //1
61037                                                     r.when_dt.format('h:ia'), //2
61038                                                     is_sub, //3
61039                                                     deactive, //4
61040                                                     desc // 5
61041                                             );
61042                                         });
61043                                     }
61044                                     d++;
61045                                 }
61046                                 
61047                                 // only do this if something added..
61048                                 if(added > 0){ 
61049                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
61050                                 }
61051                                 
61052                                 
61053                                 // push it twice. (second one with an hour..
61054                                 
61055                             }
61056                             //Roo.log(result);
61057                             this.fireEvent("load", this, o, o.request.arg);
61058                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
61059                         },
61060                     sortInfo : {field: 'when_dt', direction : 'ASC' },
61061                     proxy : {
61062                         xtype: 'HttpProxy',
61063                         xns: Roo.data,
61064                         method : 'GET',
61065                         url : baseURL + '/Roo/Shop_course.php'
61066                     },
61067                     reader : {
61068                         xtype: 'JsonReader',
61069                         xns: Roo.data,
61070                         id : 'id',
61071                         fields : [
61072                             {
61073                                 'name': 'id',
61074                                 'type': 'int'
61075                             },
61076                             {
61077                                 'name': 'when_dt',
61078                                 'type': 'string'
61079                             },
61080                             {
61081                                 'name': 'end_dt',
61082                                 'type': 'string'
61083                             },
61084                             {
61085                                 'name': 'parent_id',
61086                                 'type': 'int'
61087                             },
61088                             {
61089                                 'name': 'product_id',
61090                                 'type': 'int'
61091                             },
61092                             {
61093                                 'name': 'productitem_id',
61094                                 'type': 'int'
61095                             },
61096                             {
61097                                 'name': 'guid',
61098                                 'type': 'int'
61099                             }
61100                         ]
61101                     }
61102                 },
61103                 toolbar : {
61104                     xtype: 'Toolbar',
61105                     xns: Roo,
61106                     items : [
61107                         {
61108                             xtype: 'Button',
61109                             xns: Roo.Toolbar,
61110                             listeners : {
61111                                 click : function (_self, e)
61112                                 {
61113                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
61114                                     sd.setMonth(sd.getMonth()-1);
61115                                     _this.monthField.setValue(sd.format('Y-m-d'));
61116                                     _this.grid.ds.load({});
61117                                 }
61118                             },
61119                             text : "Back"
61120                         },
61121                         {
61122                             xtype: 'Separator',
61123                             xns: Roo.Toolbar
61124                         },
61125                         {
61126                             xtype: 'MonthField',
61127                             xns: Roo.form,
61128                             listeners : {
61129                                 render : function (_self)
61130                                 {
61131                                     _this.monthField = _self;
61132                                    // _this.monthField.set  today
61133                                 },
61134                                 select : function (combo, date)
61135                                 {
61136                                     _this.grid.ds.load({});
61137                                 }
61138                             },
61139                             value : (function() { return new Date(); })()
61140                         },
61141                         {
61142                             xtype: 'Separator',
61143                             xns: Roo.Toolbar
61144                         },
61145                         {
61146                             xtype: 'TextItem',
61147                             xns: Roo.Toolbar,
61148                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
61149                         },
61150                         {
61151                             xtype: 'Fill',
61152                             xns: Roo.Toolbar
61153                         },
61154                         {
61155                             xtype: 'Button',
61156                             xns: Roo.Toolbar,
61157                             listeners : {
61158                                 click : function (_self, e)
61159                                 {
61160                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
61161                                     sd.setMonth(sd.getMonth()+1);
61162                                     _this.monthField.setValue(sd.format('Y-m-d'));
61163                                     _this.grid.ds.load({});
61164                                 }
61165                             },
61166                             text : "Next"
61167                         }
61168                     ]
61169                 },
61170                  
61171             }
61172         };
61173         
61174         *//*
61175  * Based on:
61176  * Ext JS Library 1.1.1
61177  * Copyright(c) 2006-2007, Ext JS, LLC.
61178  *
61179  * Originally Released Under LGPL - original licence link has changed is not relivant.
61180  *
61181  * Fork - LGPL
61182  * <script type="text/javascript">
61183  */
61184  
61185 /**
61186  * @class Roo.LoadMask
61187  * A simple utility class for generically masking elements while loading data.  If the element being masked has
61188  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
61189  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
61190  * element's UpdateManager load indicator and will be destroyed after the initial load.
61191  * @constructor
61192  * Create a new LoadMask
61193  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
61194  * @param {Object} config The config object
61195  */
61196 Roo.LoadMask = function(el, config){
61197     this.el = Roo.get(el);
61198     Roo.apply(this, config);
61199     if(this.store){
61200         this.store.on('beforeload', this.onBeforeLoad, this);
61201         this.store.on('load', this.onLoad, this);
61202         this.store.on('loadexception', this.onLoadException, this);
61203         this.removeMask = false;
61204     }else{
61205         var um = this.el.getUpdateManager();
61206         um.showLoadIndicator = false; // disable the default indicator
61207         um.on('beforeupdate', this.onBeforeLoad, this);
61208         um.on('update', this.onLoad, this);
61209         um.on('failure', this.onLoad, this);
61210         this.removeMask = true;
61211     }
61212 };
61213
61214 Roo.LoadMask.prototype = {
61215     /**
61216      * @cfg {Boolean} removeMask
61217      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61218      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61219      */
61220     /**
61221      * @cfg {String} msg
61222      * The text to display in a centered loading message box (defaults to 'Loading...')
61223      */
61224     msg : 'Loading...',
61225     /**
61226      * @cfg {String} msgCls
61227      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61228      */
61229     msgCls : 'x-mask-loading',
61230
61231     /**
61232      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61233      * @type Boolean
61234      */
61235     disabled: false,
61236
61237     /**
61238      * Disables the mask to prevent it from being displayed
61239      */
61240     disable : function(){
61241        this.disabled = true;
61242     },
61243
61244     /**
61245      * Enables the mask so that it can be displayed
61246      */
61247     enable : function(){
61248         this.disabled = false;
61249     },
61250     
61251     onLoadException : function()
61252     {
61253         Roo.log(arguments);
61254         
61255         if (typeof(arguments[3]) != 'undefined') {
61256             Roo.MessageBox.alert("Error loading",arguments[3]);
61257         } 
61258         /*
61259         try {
61260             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61261                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61262             }   
61263         } catch(e) {
61264             
61265         }
61266         */
61267     
61268         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61269     },
61270     // private
61271     onLoad : function()
61272     {
61273         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61274     },
61275
61276     // private
61277     onBeforeLoad : function(){
61278         if(!this.disabled){
61279             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61280         }
61281     },
61282
61283     // private
61284     destroy : function(){
61285         if(this.store){
61286             this.store.un('beforeload', this.onBeforeLoad, this);
61287             this.store.un('load', this.onLoad, this);
61288             this.store.un('loadexception', this.onLoadException, this);
61289         }else{
61290             var um = this.el.getUpdateManager();
61291             um.un('beforeupdate', this.onBeforeLoad, this);
61292             um.un('update', this.onLoad, this);
61293             um.un('failure', this.onLoad, this);
61294         }
61295     }
61296 };/*
61297  * Based on:
61298  * Ext JS Library 1.1.1
61299  * Copyright(c) 2006-2007, Ext JS, LLC.
61300  *
61301  * Originally Released Under LGPL - original licence link has changed is not relivant.
61302  *
61303  * Fork - LGPL
61304  * <script type="text/javascript">
61305  */
61306
61307
61308 /**
61309  * @class Roo.XTemplate
61310  * @extends Roo.Template
61311  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61312 <pre><code>
61313 var t = new Roo.XTemplate(
61314         '&lt;select name="{name}"&gt;',
61315                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61316         '&lt;/select&gt;'
61317 );
61318  
61319 // then append, applying the master template values
61320  </code></pre>
61321  *
61322  * Supported features:
61323  *
61324  *  Tags:
61325
61326 <pre><code>
61327       {a_variable} - output encoded.
61328       {a_variable.format:("Y-m-d")} - call a method on the variable
61329       {a_variable:raw} - unencoded output
61330       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61331       {a_variable:this.method_on_template(...)} - call a method on the template object.
61332  
61333 </code></pre>
61334  *  The tpl tag:
61335 <pre><code>
61336         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61337         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61338         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61339         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61340   
61341         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61342         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61343 </code></pre>
61344  *      
61345  */
61346 Roo.XTemplate = function()
61347 {
61348     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61349     if (this.html) {
61350         this.compile();
61351     }
61352 };
61353
61354
61355 Roo.extend(Roo.XTemplate, Roo.Template, {
61356
61357     /**
61358      * The various sub templates
61359      */
61360     tpls : false,
61361     /**
61362      *
61363      * basic tag replacing syntax
61364      * WORD:WORD()
61365      *
61366      * // you can fake an object call by doing this
61367      *  x.t:(test,tesT) 
61368      * 
61369      */
61370     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61371
61372     /**
61373      * compile the template
61374      *
61375      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61376      *
61377      */
61378     compile: function()
61379     {
61380         var s = this.html;
61381      
61382         s = ['<tpl>', s, '</tpl>'].join('');
61383     
61384         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61385             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61386             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61387             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61388             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61389             m,
61390             id     = 0,
61391             tpls   = [];
61392     
61393         while(true == !!(m = s.match(re))){
61394             var forMatch   = m[0].match(nameRe),
61395                 ifMatch   = m[0].match(ifRe),
61396                 execMatch   = m[0].match(execRe),
61397                 namedMatch   = m[0].match(namedRe),
61398                 
61399                 exp  = null, 
61400                 fn   = null,
61401                 exec = null,
61402                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61403                 
61404             if (ifMatch) {
61405                 // if - puts fn into test..
61406                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61407                 if(exp){
61408                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61409                 }
61410             }
61411             
61412             if (execMatch) {
61413                 // exec - calls a function... returns empty if true is  returned.
61414                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61415                 if(exp){
61416                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61417                 }
61418             }
61419             
61420             
61421             if (name) {
61422                 // for = 
61423                 switch(name){
61424                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61425                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61426                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61427                 }
61428             }
61429             var uid = namedMatch ? namedMatch[1] : id;
61430             
61431             
61432             tpls.push({
61433                 id:     namedMatch ? namedMatch[1] : id,
61434                 target: name,
61435                 exec:   exec,
61436                 test:   fn,
61437                 body:   m[1] || ''
61438             });
61439             if (namedMatch) {
61440                 s = s.replace(m[0], '');
61441             } else { 
61442                 s = s.replace(m[0], '{xtpl'+ id + '}');
61443             }
61444             ++id;
61445         }
61446         this.tpls = [];
61447         for(var i = tpls.length-1; i >= 0; --i){
61448             this.compileTpl(tpls[i]);
61449             this.tpls[tpls[i].id] = tpls[i];
61450         }
61451         this.master = tpls[tpls.length-1];
61452         return this;
61453     },
61454     /**
61455      * same as applyTemplate, except it's done to one of the subTemplates
61456      * when using named templates, you can do:
61457      *
61458      * var str = pl.applySubTemplate('your-name', values);
61459      *
61460      * 
61461      * @param {Number} id of the template
61462      * @param {Object} values to apply to template
61463      * @param {Object} parent (normaly the instance of this object)
61464      */
61465     applySubTemplate : function(id, values, parent)
61466     {
61467         
61468         
61469         var t = this.tpls[id];
61470         
61471         
61472         try { 
61473             if(t.test && !t.test.call(this, values, parent)){
61474                 return '';
61475             }
61476         } catch(e) {
61477             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61478             Roo.log(e.toString());
61479             Roo.log(t.test);
61480             return ''
61481         }
61482         try { 
61483             
61484             if(t.exec && t.exec.call(this, values, parent)){
61485                 return '';
61486             }
61487         } catch(e) {
61488             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61489             Roo.log(e.toString());
61490             Roo.log(t.exec);
61491             return ''
61492         }
61493         try {
61494             var vs = t.target ? t.target.call(this, values, parent) : values;
61495             parent = t.target ? values : parent;
61496             if(t.target && vs instanceof Array){
61497                 var buf = [];
61498                 for(var i = 0, len = vs.length; i < len; i++){
61499                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61500                 }
61501                 return buf.join('');
61502             }
61503             return t.compiled.call(this, vs, parent);
61504         } catch (e) {
61505             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61506             Roo.log(e.toString());
61507             Roo.log(t.compiled);
61508             return '';
61509         }
61510     },
61511
61512     compileTpl : function(tpl)
61513     {
61514         var fm = Roo.util.Format;
61515         var useF = this.disableFormats !== true;
61516         var sep = Roo.isGecko ? "+" : ",";
61517         var undef = function(str) {
61518             Roo.log("Property not found :"  + str);
61519             return '';
61520         };
61521         
61522         var fn = function(m, name, format, args)
61523         {
61524             //Roo.log(arguments);
61525             args = args ? args.replace(/\\'/g,"'") : args;
61526             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61527             if (typeof(format) == 'undefined') {
61528                 format= 'htmlEncode';
61529             }
61530             if (format == 'raw' ) {
61531                 format = false;
61532             }
61533             
61534             if(name.substr(0, 4) == 'xtpl'){
61535                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61536             }
61537             
61538             // build an array of options to determine if value is undefined..
61539             
61540             // basically get 'xxxx.yyyy' then do
61541             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61542             //    (function () { Roo.log("Property not found"); return ''; })() :
61543             //    ......
61544             
61545             var udef_ar = [];
61546             var lookfor = '';
61547             Roo.each(name.split('.'), function(st) {
61548                 lookfor += (lookfor.length ? '.': '') + st;
61549                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61550             });
61551             
61552             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61553             
61554             
61555             if(format && useF){
61556                 
61557                 args = args ? ',' + args : "";
61558                  
61559                 if(format.substr(0, 5) != "this."){
61560                     format = "fm." + format + '(';
61561                 }else{
61562                     format = 'this.call("'+ format.substr(5) + '", ';
61563                     args = ", values";
61564                 }
61565                 
61566                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61567             }
61568              
61569             if (args.length) {
61570                 // called with xxyx.yuu:(test,test)
61571                 // change to ()
61572                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61573             }
61574             // raw.. - :raw modifier..
61575             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61576             
61577         };
61578         var body;
61579         // branched to use + in gecko and [].join() in others
61580         if(Roo.isGecko){
61581             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61582                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61583                     "';};};";
61584         }else{
61585             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61586             body.push(tpl.body.replace(/(\r\n|\n)/g,
61587                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61588             body.push("'].join('');};};");
61589             body = body.join('');
61590         }
61591         
61592         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61593        
61594         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61595         eval(body);
61596         
61597         return this;
61598     },
61599
61600     applyTemplate : function(values){
61601         return this.master.compiled.call(this, values, {});
61602         //var s = this.subs;
61603     },
61604
61605     apply : function(){
61606         return this.applyTemplate.apply(this, arguments);
61607     }
61608
61609  });
61610
61611 Roo.XTemplate.from = function(el){
61612     el = Roo.getDom(el);
61613     return new Roo.XTemplate(el.value || el.innerHTML);
61614 };