ext-all-debug.js
上传用户:shuoshiled
上传日期:2018-01-28
资源大小:10124k
文件大小:2341k
源码类别:

中间件编程

开发平台:

JavaScript

  1.             this.updateSortIcon(sortColumn, sortDir);
  2.         }
  3.     },
  4.     updateSortIcon : function(col, dir){
  5.         var sc = this.sortClasses;
  6.         var hds = this.view.innerHd.select('em').removeClass(sc);
  7.         hds.item(col).addClass(sc[dir == "DESC" ? 1 : 0]);
  8.     },
  9.     onHdClick : function(e){
  10.         var hd = e.getTarget('em', 3);
  11.         if(hd && !this.view.disableHeaders){
  12.             var index = this.view.findHeaderIndex(hd);
  13.             this.view.store.sort(this.view.columns[index].dataIndex);
  14.         }
  15.     }
  16. });/**  * @class Ext.TabPanel  * <p>A basic tab container. TabPanels can be used exactly like a standard {@link Ext.Panel}  * for layout purposes, but also have special support for containing child Components  * (<tt>{@link Ext.Container#items items}</tt>) that are managed using a  * {@link Ext.layout.CardLayout CardLayout layout manager}, and displayed as separate tabs.</p>  *  * <b>Note:</b> By default, a tab's close tool <i>destroys</i> the child tab Component  * and all its descendants. This makes the child tab Component, and all its descendants <b>unusable</b>. To enable  * re-use of a tab, configure the TabPanel with <b><code>{@link #autoDestroy autoDestroy: false}</code></b>.  *  * <p><b><u>TabPanel header/footer elements</u></b></p>  * <p>TabPanels use their {@link Ext.Panel#header header} or {@link Ext.Panel#footer footer} element  * (depending on the {@link #tabPosition} configuration) to accommodate the tab selector buttons.  * This means that a TabPanel will not display any configured title, and will not display any  * configured header {@link Ext.Panel#tools tools}.</p>  * <p>To display a header, embed the TabPanel in a {@link Ext.Panel Panel} which uses  * <b><tt>{@link Ext.Container#layout layout:'fit'}</tt></b>.</p>  *  * <p><b><u>Tab Events</u></b></p>  * <p>There is no actual tab class &mdash; each tab is simply a {@link Ext.BoxComponent Component}  * such as a {@link Ext.Panel Panel}. However, when rendered in a TabPanel, each child Component  * can fire additional events that only exist for tabs and are not available from other Components.  * These events are:</p>  * <div><ul class="mdetail-params">  * <li><tt><b>{@link Ext.Panel#activate activate}</b></tt> : Fires when this Component becomes  * the active tab.</li>  * <li><tt><b>{@link Ext.Panel#deactivate deactivate}</b></tt> : Fires when the Component that  * was the active tab becomes deactivated.</li>  * </ul></div>  * <p><b><u>Creating TabPanels from Code</u></b></p>  * <p>TabPanels can be created and rendered completely in code, as in this example:</p>  * <pre><code> var tabs = new Ext.TabPanel({     renderTo: Ext.getBody(),     activeTab: 0,     items: [{         title: 'Tab 1',         html: 'A simple tab'     },{         title: 'Tab 2',         html: 'Another one'     }] }); </code></pre>  * <p><b><u>Creating TabPanels from Existing Markup</u></b></p>  * <p>TabPanels can also be rendered from pre-existing markup in a couple of ways.</p>  * <div><ul class="mdetail-params">  *  * <li>Pre-Structured Markup</li>  * <div class="sub-desc">  * <p>A container div with one or more nested tab divs with class <tt>'x-tab'</tt> can be rendered entirely  * from existing markup (See the {@link #autoTabs} example).</p>  * </div>  *  * <li>Un-Structured Markup</li>  * <div class="sub-desc">  * <p>A TabPanel can also be rendered from markup that is not strictly structured by simply specifying by id  * which elements should be the container and the tabs. Using this method tab content can be pulled from different  * elements within the page by id regardless of page structure. For example:</p>  * <pre><code> var tabs = new Ext.TabPanel({     renderTo: 'my-tabs',     activeTab: 0,     items:[         {contentEl:'tab1', title:'Tab 1'},         {contentEl:'tab2', title:'Tab 2'}     ] }); // Note that the tabs do not have to be nested within the container (although they can be) &lt;div id="my-tabs">&lt;/div> &lt;div id="tab1" class="x-hide-display">A simple tab&lt;/div> &lt;div id="tab2" class="x-hide-display">Another one&lt;/div> </code></pre>  * Note that the tab divs in this example contain the class <tt>'x-hide-display'</tt> so that they can be rendered  * deferred without displaying outside the tabs. You could alternately set <tt>{@link #deferredRender} = false </tt>  * to render all content tabs on page load.  * </div>  *  * </ul></div>  *  * @extends Ext.Panel  * @constructor  * @param {Object} config The configuration options  * @xtype tabpanel  */ Ext.TabPanel = Ext.extend(Ext.Panel,  {     /**      * @cfg {Boolean} layoutOnTabChange      * Set to true to force a layout of the active tab when the tab is changed. Defaults to false.      * See {@link Ext.layout.CardLayout}.<code>{@link Ext.layout.CardLayout#layoutOnCardChange layoutOnCardChange}</code>.      */     /**      * @cfg {String} tabCls <b>This config option is used on <u>child Components</u> of ths TabPanel.</b> A CSS      * class name applied to the tab strip item representing the child Component, allowing special      * styling to be applied.      */     /**      * @cfg {Boolean} monitorResize True to automatically monitor window resize events and rerender the layout on      * browser resize (defaults to true).      */     monitorResize : true,     /**      * @cfg {Boolean} deferredRender      * <p><tt>true</tt> by default to defer the rendering of child <tt>{@link Ext.Container#items items}</tt>      * to the browsers DOM until a tab is activated. <tt>false</tt> will render all contained      * <tt>{@link Ext.Container#items items}</tt> as soon as the {@link Ext.layout.CardLayout layout}      * is rendered. If there is a significant amount of content or a lot of heavy controls being      * rendered into panels that are not displayed by default, setting this to <tt>true</tt> might      * improve performance.</p>      * <br><p>The <tt>deferredRender</tt> property is internally passed to the layout manager for      * TabPanels ({@link Ext.layout.CardLayout}) as its {@link Ext.layout.CardLayout#deferredRender}      * configuration value.</p>      * <br><p><b>Note</b>: leaving <tt>deferredRender</tt> as <tt>true</tt> means that the content      * within an unactivated tab will not be available. For example, this means that if the TabPanel      * is within a {@link Ext.form.FormPanel form}, then until a tab is activated, any Fields within      * unactivated tabs will not be rendered, and will therefore not be submitted and will not be      * available to either {@link Ext.form.BasicForm#getValues getValues} or      * {@link Ext.form.BasicForm#setValues setValues}.</p>      */     deferredRender : true,     /**      * @cfg {Number} tabWidth The initial width in pixels of each new tab (defaults to 120).      */     tabWidth : 120,     /**      * @cfg {Number} minTabWidth The minimum width in pixels for each tab when {@link #resizeTabs} = true (defaults to 30).      */     minTabWidth : 30,     /**      * @cfg {Boolean} resizeTabs True to automatically resize each tab so that the tabs will completely fill the      * tab strip (defaults to false).  Setting this to true may cause specific widths that might be set per tab to      * be overridden in order to fit them all into view (although {@link #minTabWidth} will always be honored).      */     resizeTabs : false,     /**      * @cfg {Boolean} enableTabScroll True to enable scrolling to tabs that may be invisible due to overflowing the      * overall TabPanel width. Only available with tabPosition:'top' (defaults to false).      */     enableTabScroll : false,     /**      * @cfg {Number} scrollIncrement The number of pixels to scroll each time a tab scroll button is pressed      * (defaults to <tt>100</tt>, or if <tt>{@link #resizeTabs} = true</tt>, the calculated tab width).  Only      * applies when <tt>{@link #enableTabScroll} = true</tt>.      */     scrollIncrement : 0,     /**      * @cfg {Number} scrollRepeatInterval Number of milliseconds between each scroll while a tab scroll button is      * continuously pressed (defaults to <tt>400</tt>).      */     scrollRepeatInterval : 400,     /**      * @cfg {Float} scrollDuration The number of milliseconds that each scroll animation should last (defaults      * to <tt>.35</tt>). Only applies when <tt>{@link #animScroll} = true</tt>.      */     scrollDuration : 0.35,     /**      * @cfg {Boolean} animScroll True to animate tab scrolling so that hidden tabs slide smoothly into view (defaults      * to <tt>true</tt>).  Only applies when <tt>{@link #enableTabScroll} = true</tt>.      */     animScroll : true,     /**      * @cfg {String} tabPosition The position where the tab strip should be rendered (defaults to <tt>'top'</tt>).      * The only other supported value is <tt>'bottom'</tt>.  <b>Note</b>: tab scrolling is only supported for      * <tt>tabPosition: 'top'</tt>.      */     tabPosition : 'top',     /**      * @cfg {String} baseCls The base CSS class applied to the panel (defaults to <tt>'x-tab-panel'</tt>).      */     baseCls : 'x-tab-panel',     /**      * @cfg {Boolean} autoTabs      * <p><tt>true</tt> to query the DOM for any divs with a class of 'x-tab' to be automatically converted      * to tabs and added to this panel (defaults to <tt>false</tt>).  Note that the query will be executed within      * the scope of the container element only (so that multiple tab panels from markup can be supported via this      * method).</p>      * <p>This method is only possible when the markup is structured correctly as a container with nested divs      * containing the class <tt>'x-tab'</tt>. To create TabPanels without these limitations, or to pull tab content      * from other elements on the page, see the example at the top of the class for generating tabs from markup.</p>      * <p>There are a couple of things to note when using this method:<ul>      * <li>When using the <tt>autoTabs</tt> config (as opposed to passing individual tab configs in the TabPanel's      * {@link #items} collection), you must use <tt>{@link #applyTo}</tt> to correctly use the specified <tt>id</tt>      * as the tab container. The <tt>autoTabs</tt> method <em>replaces</em> existing content with the TabPanel      * components.</li>      * <li>Make sure that you set <tt>{@link #deferredRender}: false</tt> so that the content elements for each      * tab will be rendered into the TabPanel immediately upon page load, otherwise they will not be transformed      * until each tab is activated and will be visible outside the TabPanel.</li>      * </ul>Example usage:</p>      * <pre><code> var tabs = new Ext.TabPanel({     applyTo: 'my-tabs',     activeTab: 0,     deferredRender: false,     autoTabs: true }); // This markup will be converted to a TabPanel from the code above &lt;div id="my-tabs">     &lt;div class="x-tab" title="Tab 1">A simple tab&lt;/div>     &lt;div class="x-tab" title="Tab 2">Another one&lt;/div> &lt;/div> </code></pre>      */     autoTabs : false,     /**      * @cfg {String} autoTabSelector The CSS selector used to search for tabs in existing markup when      * <tt>{@link #autoTabs} = true</tt> (defaults to <tt>'div.x-tab'</tt>).  This can be any valid selector      * supported by {@link Ext.DomQuery#select}. Note that the query will be executed within the scope of this      * tab panel only (so that multiple tab panels from markup can be supported on a page).      */     autoTabSelector : 'div.x-tab',     /**      * @cfg {String/Number} activeTab A string id or the numeric index of the tab that should be initially      * activated on render (defaults to none).      */     activeTab : null,     /**      * @cfg {Number} tabMargin The number of pixels of space to calculate into the sizing and scrolling of      * tabs. If you change the margin in CSS, you will need to update this value so calculations are correct      * with either <tt>{@link #resizeTabs}</tt> or scrolling tabs. (defaults to <tt>2</tt>)      */     tabMargin : 2,     /**      * @cfg {Boolean} plain </tt>true</tt> to render the tab strip without a background container image      * (defaults to <tt>false</tt>).      */     plain : false,     /**      * @cfg {Number} wheelIncrement For scrolling tabs, the number of pixels to increment on mouse wheel      * scrolling (defaults to <tt>20</tt>).      */     wheelIncrement : 20,     /*      * This is a protected property used when concatenating tab ids to the TabPanel id for internal uniqueness.      * It does not generally need to be changed, but can be if external code also uses an id scheme that can      * potentially clash with this one.      */     idDelimiter : '__',     // private     itemCls : 'x-tab-item',     // private config overrides     elements : 'body',     headerAsText : false,     frame : false,     hideBorders :true,     // private     initComponent : function(){         this.frame = false;         Ext.TabPanel.superclass.initComponent.call(this);         this.addEvents(             /**              * @event beforetabchange              * Fires before the active tab changes. Handlers can <tt>return false</tt> to cancel the tab change.              * @param {TabPanel} this              * @param {Panel} newTab The tab being activated              * @param {Panel} currentTab The current active tab              */             'beforetabchange',             /**              * @event tabchange              * Fires after the active tab has changed.              * @param {TabPanel} this              * @param {Panel} tab The new active tab              */             'tabchange',             /**              * @event contextmenu              * Relays the contextmenu event from a tab selector element in the tab strip.              * @param {TabPanel} this              * @param {Panel} tab The target tab              * @param {EventObject} e              */             'contextmenu'         );         /**          * @cfg {Object} layoutConfig          * TabPanel implicitly uses {@link Ext.layout.CardLayout} as its layout manager.          * <code>layoutConfig</code> may be used to configure this layout manager.          * <code>{@link #deferredRender}</code> and <code>{@link #layoutOnTabChange}</code>          * configured on the TabPanel will be applied as configs to the layout manager.          */         this.setLayout(new Ext.layout.CardLayout(Ext.apply({             layoutOnCardChange: this.layoutOnTabChange,             deferredRender: this.deferredRender         }, this.layoutConfig)));         if(this.tabPosition == 'top'){             this.elements += ',header';             this.stripTarget = 'header';         }else {             this.elements += ',footer';             this.stripTarget = 'footer';         }         if(!this.stack){             this.stack = Ext.TabPanel.AccessStack();         }         this.initItems();     },     // private     onRender : function(ct, position){         Ext.TabPanel.superclass.onRender.call(this, ct, position);         if(this.plain){             var pos = this.tabPosition == 'top' ? 'header' : 'footer';             this[pos].addClass('x-tab-panel-'+pos+'-plain');         }         var st = this[this.stripTarget];         this.stripWrap = st.createChild({cls:'x-tab-strip-wrap', cn:{             tag:'ul', cls:'x-tab-strip x-tab-strip-'+this.tabPosition}});         var beforeEl = (this.tabPosition=='bottom' ? this.stripWrap : null);         this.stripSpacer = st.createChild({cls:'x-tab-strip-spacer'}, beforeEl);         this.strip = new Ext.Element(this.stripWrap.dom.firstChild);         this.edge = this.strip.createChild({tag:'li', cls:'x-tab-edge'});         this.strip.createChild({cls:'x-clear'});         this.body.addClass('x-tab-panel-body-'+this.tabPosition);         /**          * @cfg {Template/XTemplate} itemTpl <p>(Optional) A {@link Ext.Template Template} or          * {@link Ext.XTemplate XTemplate} which may be provided to process the data object returned from          * <tt>{@link #getTemplateArgs}</tt> to produce a clickable selector element in the tab strip.</p>          * <p>The main element created should be a <tt>&lt;li></tt> element. In order for a click event on          * a selector element to be connected to its item, it must take its <i>id</i> from the TabPanel's          * native <tt>{@link #getTemplateArgs}</tt>.</p>          * <p>The child element which contains the title text must be marked by the CSS class          * <tt>x-tab-strip-inner</tt>.</p>          * <p>To enable closability, the created element should contain an element marked by the CSS class          * <tt>x-tab-strip-close</tt>.</p>          * <p>If a custom <tt>itemTpl</tt> is supplied, it is the developer's responsibility to create CSS          * style rules to create the desired appearance.</p>          * Below is an example of how to create customized tab selector items:<pre><code> new Ext.TabPanel({     renderTo: document.body,     minTabWidth: 115,     tabWidth: 135,     enableTabScroll: true,     width: 600,     height: 250,     defaults: {autoScroll:true},     itemTpl: new Ext.XTemplate(     '&lt;li class="{cls}" id="{id}" style="overflow:hidden">',          '&lt;tpl if="closable">',             '&lt;a class="x-tab-strip-close" onclick="return false;">&lt;/a>',          '&lt;/tpl>',          '&lt;a class="x-tab-right" href="#" onclick="return false;" style="padding-left:6px">',             '&lt;em class="x-tab-left">',                 '&lt;span class="x-tab-strip-inner">',                     '&lt;img src="{src}" style="float:left;margin:3px 3px 0 0">',                     '&lt;span style="margin-left:20px" class="x-tab-strip-text {iconCls}">{text} {extra}&lt;/span>',                 '&lt;/span>',             '&lt;/em>',         '&lt;/a>',     '&lt;/li>'     ),     getTemplateArgs: function(item) { //      Call the native method to collect the base data. Like the ID!         var result = Ext.TabPanel.prototype.getTemplateArgs.call(this, item); //      Add stuff used in our template         return Ext.apply(result, {             closable: item.closable,             src: item.iconSrc,             extra: item.extraText || ''         });     },     items: [{         title: 'New Tab 1',         iconSrc: '../shared/icons/fam/grid.png',         html: 'Tab Body 1',         closable: true     }, {         title: 'New Tab 2',         iconSrc: '../shared/icons/fam/grid.png',         html: 'Tab Body 2',         extraText: 'Extra stuff in the tab button'     }] }); </code></pre>          */         if(!this.itemTpl){             var tt = new Ext.Template(                  '<li class="{cls}" id="{id}"><a class="x-tab-strip-close" onclick="return false;"></a>',                  '<a class="x-tab-right" href="#" onclick="return false;"><em class="x-tab-left">',                  '<span class="x-tab-strip-inner"><span class="x-tab-strip-text {iconCls}">{text}</span></span>',                  '</em></a></li>'             );             tt.disableFormats = true;             tt.compile();             Ext.TabPanel.prototype.itemTpl = tt;         }         this.items.each(this.initTab, this);     },     // private     afterRender : function(){         Ext.TabPanel.superclass.afterRender.call(this);         if(this.autoTabs){             this.readTabs(false);         }         if(this.activeTab !== undefined){             var item = Ext.isObject(this.activeTab) ? this.activeTab : this.items.get(this.activeTab);             delete this.activeTab;             this.setActiveTab(item);         }     },     // private     initEvents : function(){         Ext.TabPanel.superclass.initEvents.call(this);         this.on('add', this.onAdd, this, {target: this});         this.on('remove', this.onRemove, this, {target: this});         this.mon(this.strip, 'mousedown', this.onStripMouseDown, this);         this.mon(this.strip, 'contextmenu', this.onStripContextMenu, this);         if(this.enableTabScroll){             this.mon(this.strip, 'mousewheel', this.onWheel, this);         }     },     // private     findTargets : function(e){         var item = null;         var itemEl = e.getTarget('li', this.strip);         if(itemEl){             item = this.getComponent(itemEl.id.split(this.idDelimiter)[1]);             if(item.disabled){                 return {                     close : null,                     item : null,                     el : null                 };             }         }         return {             close : e.getTarget('.x-tab-strip-close', this.strip),             item : item,             el : itemEl         };     },     // private     onStripMouseDown : function(e){         if(e.button !== 0){             return;         }         e.preventDefault();         var t = this.findTargets(e);         if(t.close){             if (t.item.fireEvent('beforeclose', t.item) !== false) {                 t.item.fireEvent('close', t.item);                 this.remove(t.item);             }             return;         }         if(t.item && t.item != this.activeTab){             this.setActiveTab(t.item);         }     },     // private     onStripContextMenu : function(e){         e.preventDefault();         var t = this.findTargets(e);         if(t.item){             this.fireEvent('contextmenu', this, t.item, e);         }     },     /**      * True to scan the markup in this tab panel for <tt>{@link #autoTabs}</tt> using the      * <tt>{@link #autoTabSelector}</tt>      * @param {Boolean} removeExisting True to remove existing tabs      */     readTabs : function(removeExisting){         if(removeExisting === true){             this.items.each(function(item){                 this.remove(item);             }, this);         }         var tabs = this.el.query(this.autoTabSelector);         for(var i = 0, len = tabs.length; i < len; i++){             var tab = tabs[i];             var title = tab.getAttribute('title');             tab.removeAttribute('title');             this.add({                 title: title,                 contentEl: tab             });         }     },     // private     initTab : function(item, index){         var before = this.strip.dom.childNodes[index];         var p = this.getTemplateArgs(item);         var el = before ?                  this.itemTpl.insertBefore(before, p) :                  this.itemTpl.append(this.strip, p);         Ext.fly(el).addClassOnOver('x-tab-strip-over');         if(item.tabTip){             Ext.fly(el).child('span.x-tab-strip-text', true).qtip = item.tabTip;         }         item.tabEl = el;         item.on('disable', this.onItemDisabled, this);         item.on('enable', this.onItemEnabled, this);         item.on('titlechange', this.onItemTitleChanged, this);         item.on('iconchange', this.onItemIconChanged, this);         item.on('beforeshow', this.onBeforeShowItem, this);     },     /**      * <p>Provides template arguments for rendering a tab selector item in the tab strip.</p>      * <p>This method returns an object hash containing properties used by the TabPanel's <tt>{@link #itemTpl}</tt>      * to create a formatted, clickable tab selector element. The properties which must be returned      * are:</p><div class="mdetail-params"><ul>      * <li><b>id</b> : String<div class="sub-desc">A unique identifier which links to the item</div></li>      * <li><b>text</b> : String<div class="sub-desc">The text to display</div></li>      * <li><b>cls</b> : String<div class="sub-desc">The CSS class name</div></li>      * <li><b>iconCls</b> : String<div class="sub-desc">A CSS class to provide appearance for an icon.</div></li>      * </ul></div>      * @param {BoxComponent} item The {@link Ext.BoxComponent BoxComponent} for which to create a selector element in the tab strip.      * @return {Object} An object hash containing the properties required to render the selector element.      */     getTemplateArgs : function(item) {         var cls = item.closable ? 'x-tab-strip-closable' : '';         if(item.disabled){             cls += ' x-item-disabled';         }         if(item.iconCls){             cls += ' x-tab-with-icon';         }         if(item.tabCls){             cls += ' ' + item.tabCls;         }         return {             id: this.id + this.idDelimiter + item.getItemId(),             text: item.title,             cls: cls,             iconCls: item.iconCls || ''         };     },     // private     onAdd : function(tp, item, index){         this.initTab(item, index);         if(this.items.getCount() == 1){             this.syncSize();         }         this.delegateUpdates();     },     // private     onBeforeAdd : function(item){         var existing = item.events ? (this.items.containsKey(item.getItemId()) ? item : null) : this.items.get(item);         if(existing){             this.setActiveTab(item);             return false;         }         Ext.TabPanel.superclass.onBeforeAdd.apply(this, arguments);         var es = item.elements;         item.elements = es ? es.replace(',header', '') : es;         item.border = (item.border === true);     },     // private     onRemove : function(tp, item){         Ext.destroy(Ext.get(this.getTabEl(item)));         this.stack.remove(item);         item.un('disable', this.onItemDisabled, this);         item.un('enable', this.onItemEnabled, this);         item.un('titlechange', this.onItemTitleChanged, this);         item.un('iconchange', this.onItemIconChanged, this);         item.un('beforeshow', this.onBeforeShowItem, this);         if(item == this.activeTab){             var next = this.stack.next();             if(next){                 this.setActiveTab(next);             }else if(this.items.getCount() > 0){                 this.setActiveTab(0);             }else{                 this.activeTab = null;             }         }         this.delegateUpdates();     },     // private     onBeforeShowItem : function(item){         if(item != this.activeTab){             this.setActiveTab(item);             return false;         }     },     // private     onItemDisabled : function(item){         var el = this.getTabEl(item);         if(el){             Ext.fly(el).addClass('x-item-disabled');         }         this.stack.remove(item);     },     // private     onItemEnabled : function(item){         var el = this.getTabEl(item);         if(el){             Ext.fly(el).removeClass('x-item-disabled');         }     },     // private     onItemTitleChanged : function(item){         var el = this.getTabEl(item);         if(el){             Ext.fly(el).child('span.x-tab-strip-text', true).innerHTML = item.title;         }     },     //private     onItemIconChanged : function(item, iconCls, oldCls){         var el = this.getTabEl(item);         if(el){             Ext.fly(el).child('span.x-tab-strip-text').replaceClass(oldCls, iconCls);         }     },     /**      * Gets the DOM element for the tab strip item which activates the child panel with the specified      * ID. Access this to change the visual treatment of the item, for example by changing the CSS class name.      * @param {Panel/Number/String} tab The tab component, or the tab's index, or the tabs id or itemId.      * @return {HTMLElement} The DOM node      */     getTabEl : function(item){         return document.getElementById(this.id + this.idDelimiter + this.getComponent(item).getItemId());     },     // private     onResize : function(){         Ext.TabPanel.superclass.onResize.apply(this, arguments);         this.delegateUpdates();     },     /**      * Suspends any internal calculations or scrolling while doing a bulk operation. See {@link #endUpdate}      */     beginUpdate : function(){         this.suspendUpdates = true;     },     /**      * Resumes calculations and scrolling at the end of a bulk operation. See {@link #beginUpdate}      */     endUpdate : function(){         this.suspendUpdates = false;         this.delegateUpdates();     },     /**      * Hides the tab strip item for the passed tab      * @param {Number/String/Panel} item The tab index, id or item      */     hideTabStripItem : function(item){         item = this.getComponent(item);         var el = this.getTabEl(item);         if(el){             el.style.display = 'none';             this.delegateUpdates();         }         this.stack.remove(item);     },     /**      * Unhides the tab strip item for the passed tab      * @param {Number/String/Panel} item The tab index, id or item      */     unhideTabStripItem : function(item){         item = this.getComponent(item);         var el = this.getTabEl(item);         if(el){             el.style.display = '';             this.delegateUpdates();         }     },     // private     delegateUpdates : function(){         if(this.suspendUpdates){             return;         }         if(this.resizeTabs && this.rendered){             this.autoSizeTabs();         }         if(this.enableTabScroll && this.rendered){             this.autoScrollTabs();         }     },     // private     autoSizeTabs : function(){         var count = this.items.length;         var ce = this.tabPosition != 'bottom' ? 'header' : 'footer';         var ow = this[ce].dom.offsetWidth;         var aw = this[ce].dom.clientWidth;         if(!this.resizeTabs || count < 1 || !aw){ // !aw for display:none             return;         }         var each = Math.max(Math.min(Math.floor((aw-4) / count) - this.tabMargin, this.tabWidth), this.minTabWidth); // -4 for float errors in IE         this.lastTabWidth = each;         var lis = this.strip.query("li:not([className^=x-tab-edge])");         for(var i = 0, len = lis.length; i < len; i++) {             var li = lis[i];             var inner = Ext.fly(li).child('.x-tab-strip-inner', true);             var tw = li.offsetWidth;             var iw = inner.offsetWidth;             inner.style.width = (each - (tw-iw)) + 'px';         }     },     // private     adjustBodyWidth : function(w){         if(this.header){             this.header.setWidth(w);         }         if(this.footer){             this.footer.setWidth(w);         }         return w;     },     /**      * Sets the specified tab as the active tab. This method fires the {@link #beforetabchange} event which      * can <tt>return false</tt> to cancel the tab change.      * @param {String/Number} item      * The id or tab Panel to activate. This parameter may be any of the following:      * <div><ul class="mdetail-params">      * <li>a <b><tt>String</tt></b> : representing the <code>{@link Ext.Component#itemId itemId}</code>      * or <code>{@link Ext.Component#id id}</code> of the child component </li>      * <li>a <b><tt>Number</tt></b> : representing the position of the child component      * within the <code>{@link Ext.Container#items items}</code> <b>property</b></li>      * </ul></div>      * <p>For additional information see {@link Ext.util.MixedCollection#get}.      */     setActiveTab : function(item){         item = this.getComponent(item);         if(!item || this.fireEvent('beforetabchange', this, item, this.activeTab) === false){             return;         }         if(!this.rendered){             this.activeTab = item;             return;         }         if(this.activeTab != item){             if(this.activeTab){                 var oldEl = this.getTabEl(this.activeTab);                 if(oldEl){                     Ext.fly(oldEl).removeClass('x-tab-strip-active');                 }                 this.activeTab.fireEvent('deactivate', this.activeTab);             }             var el = this.getTabEl(item);             Ext.fly(el).addClass('x-tab-strip-active');             this.activeTab = item;             this.stack.add(item);             this.layout.setActiveItem(item);             if(this.scrolling){                 this.scrollToTab(item, this.animScroll);             }             item.fireEvent('activate', item);             this.fireEvent('tabchange', this, item);         }     },     /**      * Gets the currently active tab.      * @return {Panel} The active tab      */     getActiveTab : function(){         return this.activeTab || null;     },     /**      * Gets the specified tab by id.      * @param {String} id The tab id      * @return {Panel} The tab      */     getItem : function(item){         return this.getComponent(item);     },     // private     autoScrollTabs : function(){         this.pos = this.tabPosition=='bottom' ? this.footer : this.header;         var count = this.items.length;         var ow = this.pos.dom.offsetWidth;         var tw = this.pos.dom.clientWidth;         var wrap = this.stripWrap;         var wd = wrap.dom;         var cw = wd.offsetWidth;         var pos = this.getScrollPos();         var l = this.edge.getOffsetsTo(this.stripWrap)[0] + pos;         if(!this.enableTabScroll || count < 1 || cw < 20){ // 20 to prevent display:none issues             return;         }         if(l <= tw){             wd.scrollLeft = 0;             wrap.setWidth(tw);             if(this.scrolling){                 this.scrolling = false;                 this.pos.removeClass('x-tab-scrolling');                 this.scrollLeft.hide();                 this.scrollRight.hide();                 // See here: http://extjs.com/forum/showthread.php?t=49308&highlight=isSafari                 if(Ext.isAir || Ext.isWebKit){                     wd.style.marginLeft = '';                     wd.style.marginRight = '';                 }             }         }else{             if(!this.scrolling){                 this.pos.addClass('x-tab-scrolling');                 // See here: http://extjs.com/forum/showthread.php?t=49308&highlight=isSafari                 if(Ext.isAir || Ext.isWebKit){                     wd.style.marginLeft = '18px';                     wd.style.marginRight = '18px';                 }             }             tw -= wrap.getMargins('lr');             wrap.setWidth(tw > 20 ? tw : 20);             if(!this.scrolling){                 if(!this.scrollLeft){                     this.createScrollers();                 }else{                     this.scrollLeft.show();                     this.scrollRight.show();                 }             }             this.scrolling = true;             if(pos > (l-tw)){ // ensure it stays within bounds                 wd.scrollLeft = l-tw;             }else{ // otherwise, make sure the active tab is still visible                 this.scrollToTab(this.activeTab, false);             }             this.updateScrollButtons();         }     },     // private     createScrollers : function(){         this.pos.addClass('x-tab-scrolling-' + this.tabPosition);         var h = this.stripWrap.dom.offsetHeight;         // left         var sl = this.pos.insertFirst({             cls:'x-tab-scroller-left'         });         sl.setHeight(h);         sl.addClassOnOver('x-tab-scroller-left-over');         this.leftRepeater = new Ext.util.ClickRepeater(sl, {             interval : this.scrollRepeatInterval,             handler: this.onScrollLeft,             scope: this         });         this.scrollLeft = sl;         // right         var sr = this.pos.insertFirst({             cls:'x-tab-scroller-right'         });         sr.setHeight(h);         sr.addClassOnOver('x-tab-scroller-right-over');         this.rightRepeater = new Ext.util.ClickRepeater(sr, {             interval : this.scrollRepeatInterval,             handler: this.onScrollRight,             scope: this         });         this.scrollRight = sr;     },     // private     getScrollWidth : function(){         return this.edge.getOffsetsTo(this.stripWrap)[0] + this.getScrollPos();     },     // private     getScrollPos : function(){         return parseInt(this.stripWrap.dom.scrollLeft, 10) || 0;     },     // private     getScrollArea : function(){         return parseInt(this.stripWrap.dom.clientWidth, 10) || 0;     },     // private     getScrollAnim : function(){         return {duration:this.scrollDuration, callback: this.updateScrollButtons, scope: this};     },     // private     getScrollIncrement : function(){         return this.scrollIncrement || (this.resizeTabs ? this.lastTabWidth+2 : 100);     },     /**      * Scrolls to a particular tab if tab scrolling is enabled      * @param {Panel} item The item to scroll to      * @param {Boolean} animate True to enable animations      */     scrollToTab : function(item, animate){         if(!item){ return; }         var el = this.getTabEl(item);         var pos = this.getScrollPos(), area = this.getScrollArea();         var left = Ext.fly(el).getOffsetsTo(this.stripWrap)[0] + pos;         var right = left + el.offsetWidth;         if(left < pos){             this.scrollTo(left, animate);         }else if(right > (pos + area)){             this.scrollTo(right - area, animate);         }     },     // private     scrollTo : function(pos, animate){         this.stripWrap.scrollTo('left', pos, animate ? this.getScrollAnim() : false);         if(!animate){             this.updateScrollButtons();         }     },     onWheel : function(e){         var d = e.getWheelDelta()*this.wheelIncrement*-1;         e.stopEvent();         var pos = this.getScrollPos();         var newpos = pos + d;         var sw = this.getScrollWidth()-this.getScrollArea();         var s = Math.max(0, Math.min(sw, newpos));         if(s != pos){             this.scrollTo(s, false);         }     },     // private     onScrollRight : function(){         var sw = this.getScrollWidth()-this.getScrollArea();         var pos = this.getScrollPos();         var s = Math.min(sw, pos + this.getScrollIncrement());         if(s != pos){             this.scrollTo(s, this.animScroll);         }     },     // private     onScrollLeft : function(){         var pos = this.getScrollPos();         var s = Math.max(0, pos - this.getScrollIncrement());         if(s != pos){             this.scrollTo(s, this.animScroll);         }     },     // private     updateScrollButtons : function(){         var pos = this.getScrollPos();         this.scrollLeft[pos === 0 ? 'addClass' : 'removeClass']('x-tab-scroller-left-disabled');         this.scrollRight[pos >= (this.getScrollWidth()-this.getScrollArea()) ? 'addClass' : 'removeClass']('x-tab-scroller-right-disabled');     },     // private     beforeDestroy : function() {         if(this.items){             this.items.each(function(item){                 if(item && item.tabEl){                     Ext.get(item.tabEl).removeAllListeners();                     item.tabEl = null;                 }             }, this);         }         if(this.strip){             this.strip.removeAllListeners();         }         Ext.TabPanel.superclass.beforeDestroy.apply(this);     }     /**      * @cfg {Boolean} collapsible      * @hide      */     /**      * @cfg {String} header      * @hide      */     /**      * @cfg {Boolean} headerAsText      * @hide      */     /**      * @property header      * @hide      */     /**      * @property title      * @hide      */     /**      * @cfg {Array} tools      * @hide      */     /**      * @cfg {Array} toolTemplate      * @hide      */     /**      * @cfg {Boolean} hideCollapseTool      * @hide      */     /**      * @cfg {Boolean} titleCollapse      * @hide      */     /**      * @cfg {Boolean} collapsed      * @hide      */     /**      * @cfg {String} layout      * @hide      */     /**      * @cfg {Boolean} preventBodyReset      * @hide      */ }); Ext.reg('tabpanel', Ext.TabPanel); /**  * See {@link #setActiveTab}. Sets the specified tab as the active tab. This method fires  * the {@link #beforetabchange} event which can <tt>return false</tt> to cancel the tab change.  * @param {String/Panel} tab The id or tab Panel to activate  * @method activate  */ Ext.TabPanel.prototype.activate = Ext.TabPanel.prototype.setActiveTab; // private utility class used by TabPanel Ext.TabPanel.AccessStack = function(){     var items = [];     return {         add : function(item){             items.push(item);             if(items.length > 10){                 items.shift();             }         },         remove : function(item){             var s = [];             for(var i = 0, len = items.length; i < len; i++) {                 if(items[i] != item){                     s.push(items[i]);                 }             }             items = s;         },         next : function(){             return items.pop();         }     }; };/**  * @class Ext.Button  * @extends Ext.BoxComponent  * Simple Button class  * @cfg {String} text The button text to be used as innerHTML (html tags are accepted)  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:'x-btn-text-icon')  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event).  * The handler is passed the following parameters:<div class="mdetail-params"><ul>  * <li><code>b</code> : Button<div class="sub-desc">This Button.</div></li>  * <li><code>e</code> : EventObject<div class="sub-desc">The click event.</div></li>  * </ul></div>  * @cfg {Object} scope The scope (<tt><b>this</b></tt> reference) in which the handler is executed. Defaults to this Button.  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width).  * See also {@link Ext.Panel}.<tt>{@link Ext.Panel#minButtonWidth minButtonWidth}</tt>.  * @cfg {String/Object} tooltip The tooltip for the button - can be a string to be used as innerHTML (html tags are accepted) or QuickTips config object  * @cfg {Boolean} hidden True to start hidden (defaults to false)  * @cfg {Boolean} disabled True to start disabled (defaults to false)  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed)  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be  * a {@link Ext.util.ClickRepeater ClickRepeater} config object (defaults to false).  * @constructor  * Create a new button  * @param {Object} config The config object  * @xtype button  */ Ext.Button = Ext.extend(Ext.BoxComponent, {     /**      * Read-only. True if this button is hidden      * @type Boolean      */     hidden : false,     /**      * Read-only. True if this button is disabled      * @type Boolean      */     disabled : false,     /**      * Read-only. True if this button is pressed (only if enableToggle = true)      * @type Boolean      */     pressed : false,     /**      * The Button's owner {@link Ext.Panel} (defaults to undefined, and is set automatically when      * the Button is added to a container).  Read-only.      * @type Ext.Panel      * @property ownerCt      */     /**      * @cfg {Number} tabIndex Set a DOM tabIndex for this button (defaults to undefined)      */     /**      * @cfg {Boolean} allowDepress      * False to not allow a pressed Button to be depressed (defaults to undefined). Only valid when {@link #enableToggle} is true.      */     /**      * @cfg {Boolean} enableToggle      * True to enable pressed/not pressed toggling (defaults to false)      */     enableToggle: false,     /**      * @cfg {Function} toggleHandler      * Function called when a Button with {@link #enableToggle} set to true is clicked. Two arguments are passed:<ul class="mdetail-params">      * <li><b>button</b> : Ext.Button<div class="sub-desc">this Button object</div></li>      * <li><b>state</b> : Boolean<div class="sub-desc">The next state if the Button, true means pressed.</div></li>      * </ul>      */     /**      * @cfg {Mixed} menu      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).      */     /**      * @cfg {String} menuAlign      * The position to align the menu to (see {@link Ext.Element#alignTo} for more details, defaults to 'tl-bl?').      */     menuAlign : 'tl-bl?',     /**      * @cfg {String} overflowText If used in a {@link Ext.Toolbar Toolbar}, the      * text to be used if this item is shown in the overflow menu. See also      * {@link Ext.Toolbar.Item}.<code>{@link Ext.Toolbar.Item#overflowText overflowText}</code>.      */     /**      * @cfg {String} iconCls      * A css class which sets a background image to be used as the icon for this button      */     /**      * @cfg {String} type      * submit, reset or button - defaults to 'button'      */     type : 'button',     // private     menuClassTarget: 'tr:nth(2)',     /**      * @cfg {String} clickEvent      * The type of event to map to the button's event handler (defaults to 'click')      */     clickEvent : 'click',     /**      * @cfg {Boolean} handleMouseEvents      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)      */     handleMouseEvents : true,     /**      * @cfg {String} tooltipType      * The type of tooltip to use. Either 'qtip' (default) for QuickTips or 'title' for title attribute.      */     tooltipType : 'qtip',     /**      * @cfg {String} buttonSelector      * <p>(Optional) A {@link Ext.DomQuery DomQuery} selector which is used to extract the active, clickable element from the      * DOM structure created.</p>      * <p>When a custom {@link #template} is used, you  must ensure that this selector results in the selection of      * a focussable element.</p>      * <p>Defaults to <b><tt>"button:first-child"</tt></b>.</p>      */     buttonSelector : 'button:first-child',     /**      * @cfg {String} scale      * <p>(Optional) The size of the Button. Three values are allowed:</p>      * <ul class="mdetail-params">      * <li>'small'<div class="sub-desc">Results in the button element being 16px high.</div></li>      * <li>'medium'<div class="sub-desc">Results in the button element being 24px high.</div></li>      * <li>'large'<div class="sub-desc">Results in the button element being 32px high.</div></li>      * </ul>      * <p>Defaults to <b><tt>'small'</tt></b>.</p>      */     scale: 'small',     /**      * @cfg {String} iconAlign      * <p>(Optional) The side of the Button box to render the icon. Four values are allowed:</p>      * <ul class="mdetail-params">      * <li>'top'<div class="sub-desc"></div></li>      * <li>'right'<div class="sub-desc"></div></li>      * <li>'bottom'<div class="sub-desc"></div></li>      * <li>'left'<div class="sub-desc"></div></li>      * </ul>      * <p>Defaults to <b><tt>'left'</tt></b>.</p>      */     iconAlign : 'left',     /**      * @cfg {String} arrowAlign      * <p>(Optional) The side of the Button box to render the arrow if the button has an associated {@link #menu}.      * Two values are allowed:</p>      * <ul class="mdetail-params">      * <li>'right'<div class="sub-desc"></div></li>      * <li>'bottom'<div class="sub-desc"></div></li>      * </ul>      * <p>Defaults to <b><tt>'right'</tt></b>.</p>      */     arrowAlign : 'right',     /**      * @cfg {Ext.Template} template (Optional)      * <p>A {@link Ext.Template Template} used to create the Button's DOM structure.</p>      * Instances, or subclasses which need a different DOM structure may provide a different      * template layout in conjunction with an implementation of {@link #getTemplateArgs}.      * @type Ext.Template      * @property template      */     /**      * @cfg {String} cls      * A CSS class string to apply to the button's main element.      */     /**      * @property menu      * @type Menu      * The {@link Ext.menu.Menu Menu} object associated with this Button when configured with the {@link #menu} config option.      */     initComponent : function(){         Ext.Button.superclass.initComponent.call(this);         this.addEvents(             /**              * @event click              * Fires when this button is clicked              * @param {Button} this              * @param {EventObject} e The click event              */             'click',             /**              * @event toggle              * Fires when the 'pressed' state of this button changes (only if enableToggle = true)              * @param {Button} this              * @param {Boolean} pressed              */             'toggle',             /**              * @event mouseover              * Fires when the mouse hovers over the button              * @param {Button} this              * @param {Event} e The event object              */             'mouseover',             /**              * @event mouseout              * Fires when the mouse exits the button              * @param {Button} this              * @param {Event} e The event object              */             'mouseout',             /**              * @event menushow              * If this button has a menu, this event fires when it is shown              * @param {Button} this              * @param {Menu} menu              */             'menushow',             /**              * @event menuhide              * If this button has a menu, this event fires when it is hidden              * @param {Button} this              * @param {Menu} menu              */             'menuhide',             /**              * @event menutriggerover              * If this button has a menu, this event fires when the mouse enters the menu triggering element              * @param {Button} this              * @param {Menu} menu              * @param {EventObject} e              */             'menutriggerover',             /**              * @event menutriggerout              * If this button has a menu, this event fires when the mouse leaves the menu triggering element              * @param {Button} this              * @param {Menu} menu              * @param {EventObject} e              */             'menutriggerout'         );         if(this.menu){             this.menu = Ext.menu.MenuMgr.get(this.menu);         }         if(Ext.isString(this.toggleGroup)){             this.enableToggle = true;         }     }, /**   * <p>This method returns an object which provides substitution parameters for the {@link #template Template} used   * to create this Button's DOM structure.</p>   * <p>Instances or subclasses which use a different Template to create a different DOM structure may need to provide their   * own implementation of this method.</p>   * <p>The default implementation which provides data for the default {@link #template} returns an Array containing the   * following items:</p><div class="mdetail-params"><ul>   * <li>The Button's {@link #text}</li>   * <li>The &lt;button&gt;'s {@link #type}</li>   * <li>The {@link iconCls} applied to the &lt;button&gt; {@link #btnEl element}</li>   * <li>The {@link #cls} applied to the Button's main {@link #getEl Element}</li>   * <li>A CSS class name controlling the Button's {@link #scale} and {@link #iconAlign icon alignment}</li>   * <li>A CSS class name which applies an arrow to the Button if configured with a {@link #menu}</li>   * </ul></div>   * @return {Object} Substitution data for a Template.  */     getTemplateArgs : function(){         var cls = (this.cls || '');         cls += (this.iconCls || this.icon) ? (this.text ? ' x-btn-text-icon' : ' x-btn-icon') : ' x-btn-noicon';         if(this.pressed){             cls += ' x-btn-pressed';         }         return [this.text || '&#160;', this.type, this.iconCls || '', cls, 'x-btn-' + this.scale + ' x-btn-icon-' + this.scale + '-' + this.iconAlign, this.getMenuClass()];     },     // protected     getMenuClass : function(){         return this.menu ? (this.arrowAlign != 'bottom' ? 'x-btn-arrow' : 'x-btn-arrow-bottom') : '';     },     // private     onRender : function(ct, position){         if(!this.template){             if(!Ext.Button.buttonTemplate){                 // hideous table template                 Ext.Button.buttonTemplate = new Ext.Template(                     '<table cellspacing="0" class="x-btn {3}"><tbody class="{4}">',                     '<tr><td class="x-btn-tl"><i>&#160;</i></td><td class="x-btn-tc"></td><td class="x-btn-tr"><i>&#160;</i></td></tr>',                     '<tr><td class="x-btn-ml"><i>&#160;</i></td><td class="x-btn-mc"><em class="{5}" unselectable="on"><button class="x-btn-text {2}" type="{1}">{0}</button></em></td><td class="x-btn-mr"><i>&#160;</i></td></tr>',                     '<tr><td class="x-btn-bl"><i>&#160;</i></td><td class="x-btn-bc"></td><td class="x-btn-br"><i>&#160;</i></td></tr>',                     "</tbody></table>");                 Ext.Button.buttonTemplate.compile();             }             this.template = Ext.Button.buttonTemplate;         }         var btn, targs = this.getTemplateArgs();         if(position){             btn = this.template.insertBefore(position, targs, true);         }else{             btn = this.template.append(ct, targs, true);         }         /**          * An {@link Ext.Element Element} encapsulating the Button's clickable element. By default,          * this references a <tt>&lt;button&gt;</tt> element. Read only.          * @type Ext.Element          * @property btnEl          */         this.btnEl = btn.child(this.buttonSelector);         this.mon(this.btnEl, {             scope: this,             focus: this.onFocus,             blur: this.onBlur         });         this.initButtonEl(btn, this.btnEl);         Ext.ButtonToggleMgr.register(this);     },     // private     initButtonEl : function(btn, btnEl){         this.el = btn;         if(this.id){             this.el.dom.id = this.el.id = this.id;         }         if(this.icon){             btnEl.setStyle('background-image', 'url(' +this.icon +')');         }         if(this.tabIndex !== undefined){             btnEl.dom.tabIndex = this.tabIndex;         }         if(this.tooltip){             this.setTooltip(this.tooltip, true);         }         if(this.handleMouseEvents){             this.mon(btn, {                 scope: this,                 mouseover: this.onMouseOver,                 mousedown: this.onMouseDown             });                          // new functionality for monitoring on the document level             //this.mon(btn, 'mouseout', this.onMouseOut, this);         }         if(this.menu){             this.mon(this.menu, {                 scope: this,                 show: this.onMenuShow,                 hide: this.onMenuHide             });         }         if(this.repeat){             var repeater = new Ext.util.ClickRepeater(btn, Ext.isObject(this.repeat) ? this.repeat : {});             this.mon(repeater, 'click', this.onClick, this);         }                  this.mon(btn, this.clickEvent, this.onClick, this);     },     // private     afterRender : function(){         Ext.Button.superclass.afterRender.call(this);         this.doAutoWidth();     },     /**      * Sets the CSS class that provides a background image to use as the button's icon.  This method also changes      * the value of the {@link iconCls} config internally.      * @param {String} cls The CSS class providing the icon image      * @return {Ext.Button} this      */     setIconClass : function(cls){         if(this.el){             this.btnEl.replaceClass(this.iconCls, cls);         }         this.iconCls = cls;         return this;     },     /**      * Sets the tooltip for this Button.      * @param {String/Object} tooltip. This may be:<div class="mdesc-details"><ul>      * <li><b>String</b> : A string to be used as innerHTML (html tags are accepted) to show in a tooltip</li>      * <li><b>Object</b> : A configuration object for {@link Ext.QuickTips#register}.</li>      * </ul></div>      * @return {Ext.Button} this      */     setTooltip : function(tooltip, /* private */ initial){         if(this.rendered){             if(!initial){                 this.clearTip();             }             if(Ext.isObject(tooltip)){                 Ext.QuickTips.register(Ext.apply({                       target: this.btnEl.id                 }, tooltip));                 this.tooltip = tooltip;             }else{                 this.btnEl.dom[this.tooltipType] = tooltip;             }         }else{             this.tooltip = tooltip;         }         return this;     },          // private     clearTip: function(){         if(Ext.isObject(this.tooltip)){             Ext.QuickTips.unregister(this.btnEl);         }     },          // private     beforeDestroy: function(){         if(this.rendered){             this.clearTip();         }         Ext.destroy(this.menu, this.repeater);     },     // private     onDestroy : function(){         var doc = Ext.getDoc();         doc.un('mouseover', this.monitorMouseOver, this);         doc.un('mouseup', this.onMouseUp, this);         if(this.rendered){             Ext.ButtonToggleMgr.unregister(this);         }     },     // private     doAutoWidth : function(){         if(this.el && this.text && this.width === undefined){             this.el.setWidth('auto');             if(Ext.isIE7 && Ext.isStrict){                 var ib = this.btnEl;                 if(ib && ib.getWidth() > 20){                     ib.clip();                     ib.setWidth(Ext.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));                 }             }             if(this.minWidth){                 if(this.el.getWidth() < this.minWidth){                     this.el.setWidth(this.minWidth);                 }             }         }     },     /**      * Assigns this Button's click handler      * @param {Function} handler The function to call when the button is clicked      * @param {Object} scope (optional) Scope for the function passed in      * @return {Ext.Button} this      */     setHandler : function(handler, scope){         this.handler = handler;         this.scope = scope;         return this;     },     /**      * Sets this Button's text      * @param {String} text The button text      * @return {Ext.Button} this      */     setText : function(text){         this.text = text;         if(this.el){             this.el.child('td.x-btn-mc ' + this.buttonSelector).update(text);         }         this.doAutoWidth();         return this;     },     /**      * Gets the text for this Button      * @return {String} The button text      */     getText : function(){         return this.text;     },     /**      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.      * @param {Boolean} state (optional) Force a particular state      * @param {Boolean} supressEvent (optional) True to stop events being fired when calling this method.      * @return {Ext.Button} this      */     toggle : function(state, suppressEvent){         state = state === undefined ? !this.pressed : !!state;         if(state != this.pressed){             this.el[state ? 'addClass' : 'removeClass']('x-btn-pressed');             this.pressed = state;             if(!suppressEvent){                 this.fireEvent('toggle', this, state);                 if(this.toggleHandler){                     this.toggleHandler.call(this.scope || this, this, state);                 }             }         }         return this;     },     /**      * Focus the button      */     focus : function(){         this.btnEl.focus();     },     // private     onDisable : function(){         this.onDisableChange(true);     },     // private     onEnable : function(){         this.onDisableChange(false);     },          onDisableChange : function(disabled){         if(this.el){             if(!Ext.isIE6 || !this.text){                 this.el[disabled ? 'addClass' : 'removeClass'](this.disabledClass);             }             this.el.dom.disabled = disabled;         }         this.disabled = disabled;     },     /**      * Show this button's menu (if it has one)      */     showMenu : function(){         if(this.rendered && this.menu){             if(this.tooltip){                 Ext.QuickTips.getQuickTip().cancelShow(this.btnEl);             }             this.menu.show(this.el, this.menuAlign);         }         return this;     },     /**      * Hide this button's menu (if it has one)      */     hideMenu : function(){         if(this.menu){             this.menu.hide();         }         return this;     },     /**      * Returns true if the button has a menu and it is visible      * @return {Boolean}      */     hasVisibleMenu : function(){         return this.menu && this.menu.isVisible();     },     // private     onClick : function(e){         if(e){             e.preventDefault();         }         if(e.button !== 0){             return;         }         if(!this.disabled){             if(this.enableToggle && (this.allowDepress !== false || !this.pressed)){                 this.toggle();             }             if(this.menu && !this.menu.isVisible() && !this.ignoreNextClick){                 this.showMenu();             }             this.fireEvent('click', this, e);             if(this.handler){                 //this.el.removeClass('x-btn-over');                 this.handler.call(this.scope || this, this, e);             }         }     },     // private     isMenuTriggerOver : function(e, internal){         return this.menu && !internal;     },     // private     isMenuTriggerOut : function(e, internal){         return this.menu && !internal;     },     // private     onMouseOver : function(e){         if(!this.disabled){             var internal = e.within(this.el,  true);             if(!internal){                 this.el.addClass('x-btn-over');                 if(!this.monitoringMouseOver){                     Ext.getDoc().on('mouseover', this.monitorMouseOver, this);                     this.monitoringMouseOver = true;                 }                 this.fireEvent('mouseover', this, e);             }             if(this.isMenuTriggerOver(e, internal)){                 this.fireEvent('menutriggerover', this, this.menu, e);             }         }     },     // private     monitorMouseOver : function(e){         if(e.target != this.el.dom && !e.within(this.el)){             if(this.monitoringMouseOver){                 Ext.getDoc().un('mouseover', this.monitorMouseOver, this);                 this.monitoringMouseOver = false;             }             this.onMouseOut(e);         }     },     // private     onMouseOut : function(e){         var internal = e.within(this.el) && e.target != this.el.dom;         this.el.removeClass('x-btn-over');         this.fireEvent('mouseout', this, e);         if(this.isMenuTriggerOut(e, internal)){             this.fireEvent('menutriggerout', this, this.menu, e);         }     },     // private     onFocus : function(e){         if(!this.disabled){             this.el.addClass('x-btn-focus');         }     },     // private     onBlur : function(e){         this.el.removeClass('x-btn-focus');     },     // private     getClickEl : function(e, isUp){        return this.el;     },     // private     onMouseDown : function(e){         if(!this.disabled && e.button === 0){             this.getClickEl(e).addClass('x-btn-click');             Ext.getDoc().on('mouseup', this.onMouseUp, this);         }     },     // private     onMouseUp : function(e){         if(e.button === 0){             this.getClickEl(e, true).removeClass('x-btn-click');             Ext.getDoc().un('mouseup', this.onMouseUp, this);         }     },     // private     onMenuShow : function(e){         this.ignoreNextClick = 0;         this.el.addClass('x-btn-menu-active');         this.fireEvent('menushow', this, this.menu);     },     // private     onMenuHide : function(e){         this.el.removeClass('x-btn-menu-active');         this.ignoreNextClick = this.restoreClick.defer(250, this);         this.fireEvent('menuhide', this, this.menu);     },     // private     restoreClick : function(){         this.ignoreNextClick = 0;     }     /**      * @cfg {String} autoEl @hide      */ }); Ext.reg('button', Ext.Button); // Private utility class used by Button Ext.ButtonToggleMgr = function(){    var groups = {};    function toggleGroup(btn, state){        if(state){            var g = groups[btn.toggleGroup];            for(var i = 0, l = g.length; i < l; i++){                if(g[i] != btn){                    g[i].toggle(false);                }            }        }    }    return {        register : function(btn){            if(!btn.toggleGroup){                return;            }            var g = groups[btn.toggleGroup];            if(!g){                g = groups[btn.toggleGroup] = [];            }            g.push(btn);            btn.on('toggle', toggleGroup);        },        unregister : function(btn){            if(!btn.toggleGroup){                return;            }            var g = groups[btn.toggleGroup];            if(g){                g.remove(btn);                btn.un('toggle', toggleGroup);            }        },        /**         * Gets the pressed button in the passed group or null         * @param {String} group         * @return Button         */        getPressed : function(group){            var g = groups[group];            if(g){                for(var i = 0, len = g.length; i < len; i++){                    if(g[i].pressed === true){                        return g[i];                    }                }            }            return null;        }    }; }();/**
  17.  * @class Ext.SplitButton
  18.  * @extends Ext.Button
  19.  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
  20.  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
  21.  * options to the primary button action, but any custom handler can provide the arrowclick implementation.  Example usage:
  22.  * <pre><code>
  23. // display a dropdown menu:
  24. new Ext.SplitButton({
  25. renderTo: 'button-ct', // the container id
  26.     text: 'Options',
  27.     handler: optionsHandler, // handle a click on the button itself
  28.     menu: new Ext.menu.Menu({
  29.         items: [
  30.          // these items will render as dropdown menu items when the arrow is clicked:
  31.         {text: 'Item 1', handler: item1Handler},
  32.         {text: 'Item 2', handler: item2Handler}
  33.         ]
  34.     })
  35. });
  36. // Instead of showing a menu, you provide any type of custom
  37. // functionality you want when the dropdown arrow is clicked:
  38. new Ext.SplitButton({
  39. renderTo: 'button-ct',
  40.     text: 'Options',
  41.     handler: optionsHandler,
  42.     arrowHandler: myCustomHandler
  43. });
  44. </code></pre>
  45.  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
  46.  * @cfg {String} arrowTooltip The title attribute of the arrow
  47.  * @constructor
  48.  * Create a new menu button
  49.  * @param {Object} config The config object
  50.  * @xtype splitbutton
  51.  */
  52. Ext.SplitButton = Ext.extend(Ext.Button, {
  53. // private
  54.     arrowSelector : 'em',
  55.     split: true,
  56.     // private
  57.     initComponent : function(){
  58.         Ext.SplitButton.superclass.initComponent.call(this);
  59.         /**
  60.          * @event arrowclick
  61.          * Fires when this button's arrow is clicked
  62.          * @param {MenuButton} this
  63.          * @param {EventObject} e The click event
  64.          */
  65.         this.addEvents("arrowclick");
  66.     },
  67.     // private
  68.     onRender : function(){
  69.         Ext.SplitButton.superclass.onRender.apply(this, arguments);
  70.         if(this.arrowTooltip){
  71.             this.el.child(this.arrowSelector).dom[this.tooltipType] = this.arrowTooltip;
  72.         }
  73.     },
  74.     /**
  75.      * Sets this button's arrow click handler.
  76.      * @param {Function} handler The function to call when the arrow is clicked
  77.      * @param {Object} scope (optional) Scope for the function passed above
  78.      */
  79.     setArrowHandler : function(handler, scope){
  80.         this.arrowHandler = handler;
  81.         this.scope = scope;
  82.     },
  83.     getMenuClass : function(){
  84.         return 'x-btn-split' + (this.arrowAlign == 'bottom' ? '-bottom' : '');
  85.     },
  86.     isClickOnArrow : function(e){
  87.         return this.arrowAlign != 'bottom' ?
  88.                e.getPageX() > this.el.child(this.buttonSelector).getRegion().right :
  89.                e.getPageY() > this.el.child(this.buttonSelector).getRegion().bottom;
  90.     },
  91.     // private
  92.     onClick : function(e, t){
  93.         e.preventDefault();
  94.         if(!this.disabled){
  95.             if(this.isClickOnArrow(e)){
  96.                 if(this.menu && !this.menu.isVisible() && !this.ignoreNextClick){
  97.                     this.showMenu();
  98.                 }
  99.                 this.fireEvent("arrowclick", this, e);
  100.                 if(this.arrowHandler){
  101.                     this.arrowHandler.call(this.scope || this, this, e);
  102.                 }
  103.             }else{
  104.                 if(this.enableToggle){
  105.                     this.toggle();
  106.                 }
  107.                 this.fireEvent("click", this, e);
  108.                 if(this.handler){
  109.                     this.handler.call(this.scope || this, this, e);
  110.                 }
  111.             }
  112.         }
  113.     },
  114.     // private
  115.     isMenuTriggerOver : function(e){
  116.         return this.menu && e.target.tagName == 'em';
  117.     },
  118.     // private
  119.     isMenuTriggerOut : function(e, internal){
  120.         return this.menu && e.target.tagName != 'em';
  121.     }
  122. });
  123. Ext.reg('splitbutton', Ext.SplitButton);/**
  124.  * @class Ext.CycleButton
  125.  * @extends Ext.SplitButton
  126.  * A specialized SplitButton that contains a menu of {@link Ext.menu.CheckItem} elements.  The button automatically
  127.  * cycles through each menu item on click, raising the button's {@link #change} event (or calling the button's
  128.  * {@link #changeHandler} function, if supplied) for the active menu item. Clicking on the arrow section of the
  129.  * button displays the dropdown menu just like a normal SplitButton.  Example usage:
  130.  * <pre><code>
  131. var btn = new Ext.CycleButton({
  132.     showText: true,
  133.     prependText: 'View as ',
  134.     items: [{
  135.         text:'text only',
  136.         iconCls:'view-text',
  137.         checked:true
  138.     },{
  139.         text:'HTML',
  140.         iconCls:'view-html'
  141.     }],
  142.     changeHandler:function(btn, item){
  143.         Ext.Msg.alert('Change View', item.text);
  144.     }
  145. });
  146. </code></pre>
  147.  * @constructor
  148.  * Create a new split button
  149.  * @param {Object} config The config object
  150.  * @xtype cycle
  151.  */
  152. Ext.CycleButton = Ext.extend(Ext.SplitButton, {
  153.     /**
  154.      * @cfg {Array} items An array of {@link Ext.menu.CheckItem} <b>config</b> objects to be used when creating the
  155.      * button's menu items (e.g., {text:'Foo', iconCls:'foo-icon'})
  156.      */
  157.     /**
  158.      * @cfg {Boolean} showText True to display the active item's text as the button text (defaults to false)
  159.      */
  160.     /**
  161.      * @cfg {String} prependText A static string to prepend before the active item's text when displayed as the
  162.      * button's text (only applies when showText = true, defaults to '')
  163.      */
  164.     /**
  165.      * @cfg {Function} changeHandler A callback function that will be invoked each time the active menu
  166.      * item in the button's menu has changed.  If this callback is not supplied, the SplitButton will instead
  167.      * fire the {@link #change} event on active item change.  The changeHandler function will be called with the
  168.      * following argument list: (SplitButton this, Ext.menu.CheckItem item)
  169.      */
  170.     /**
  171.      * @cfg {String} forceIcon A css class which sets an image to be used as the static icon for this button.  This
  172.      * icon will always be displayed regardless of which item is selected in the dropdown list.  This overrides the 
  173.      * default behavior of changing the button's icon to match the selected item's icon on change.
  174.      */
  175.     /**
  176.      * @property menu
  177.      * @type Menu
  178.      * The {@link Ext.menu.Menu Menu} object used to display the {@link Ext.menu.CheckItem CheckItems} representing the available choices.
  179.      */
  180.     // private
  181.     getItemText : function(item){
  182.         if(item && this.showText === true){
  183.             var text = '';
  184.             if(this.prependText){
  185.                 text += this.prependText;
  186.             }
  187.             text += item.text;
  188.             return text;
  189.         }
  190.         return undefined;
  191.     },
  192.     /**
  193.      * Sets the button's active menu item.
  194.      * @param {Ext.menu.CheckItem} item The item to activate
  195.      * @param {Boolean} suppressEvent True to prevent the button's change event from firing (defaults to false)
  196.      */
  197.     setActiveItem : function(item, suppressEvent){
  198.         if(typeof item != 'object'){
  199.             item = this.menu.items.get(item);
  200.         }
  201.         if(item){
  202.             if(!this.rendered){
  203.                 this.text = this.getItemText(item);
  204.                 this.iconCls = item.iconCls;
  205.             }else{
  206.                 var t = this.getItemText(item);
  207.                 if(t){
  208.                     this.setText(t);
  209.                 }
  210.                 this.setIconClass(item.iconCls);
  211.             }
  212.             this.activeItem = item;
  213.             if(!item.checked){
  214.                 item.setChecked(true, true);
  215.             }
  216.             if(this.forceIcon){
  217.                 this.setIconClass(this.forceIcon);
  218.             }
  219.             if(!suppressEvent){
  220.                 this.fireEvent('change', this, item);
  221.             }
  222.         }
  223.     },
  224.     /**
  225.      * Gets the currently active menu item.
  226.      * @return {Ext.menu.CheckItem} The active item
  227.      */
  228.     getActiveItem : function(){
  229.         return this.activeItem;
  230.     },
  231.     // private
  232.     initComponent : function(){
  233.         this.addEvents(
  234.             /**
  235.              * @event change
  236.              * Fires after the button's active menu item has changed.  Note that if a {@link #changeHandler} function
  237.              * is set on this CycleButton, it will be called instead on active item change and this change event will
  238.              * not be fired.
  239.              * @param {Ext.CycleButton} this
  240.              * @param {Ext.menu.CheckItem} item The menu item that was selected
  241.              */
  242.             "change"
  243.         );
  244.         if(this.changeHandler){
  245.             this.on('change', this.changeHandler, this.scope||this);
  246.             delete this.changeHandler;
  247.         }
  248.         this.itemCount = this.items.length;
  249.         this.menu = {cls:'x-cycle-menu', items:[]};
  250.         var checked;
  251.         for(var i = 0, len = this.itemCount; i < len; i++){
  252.             var item = this.items[i];
  253.             item.group = item.group || this.id;
  254.             item.itemIndex = i;
  255.             item.checkHandler = this.checkHandler;
  256.             item.scope = this;
  257.             item.checked = item.checked || false;
  258.             this.menu.items.push(item);
  259.             if(item.checked){
  260.                 checked = item;
  261.             }
  262.         }
  263.         this.setActiveItem(checked, true);
  264.         Ext.CycleButton.superclass.initComponent.call(this);
  265.         this.on('click', this.toggleSelected, this);
  266.     },
  267.     // private
  268.     checkHandler : function(item, pressed){
  269.         if(pressed){
  270.             this.setActiveItem(item);
  271.         }
  272.     },
  273.     /**
  274.      * This is normally called internally on button click, but can be called externally to advance the button's
  275.      * active item programmatically to the next one in the menu.  If the current item is the last one in the menu
  276.      * the active item will be set to the first item in the menu.
  277.      */
  278.     toggleSelected : function(){
  279.         this.menu.render();
  280.         
  281.         var nextIdx, checkItem;
  282.         for (var i = 1; i < this.itemCount; i++) {
  283.             nextIdx = (this.activeItem.itemIndex + i) % this.itemCount;
  284.             // check the potential item
  285.             checkItem = this.menu.items.itemAt(nextIdx);
  286.             // if its not disabled then check it.
  287.             if (!checkItem.disabled) {
  288.                 checkItem.setChecked(true);
  289.                 break;
  290.             }
  291.         }
  292.     }
  293. });
  294. Ext.reg('cycle', Ext.CycleButton);/**
  295.  * @class Ext.layout.ToolbarLayout
  296.  * @extends Ext.layout.ContainerLayout
  297.  * Layout manager implicitly used by Ext.Toolbar.
  298.  */
  299. Ext.layout.ToolbarLayout = Ext.extend(Ext.layout.ContainerLayout, {
  300.     monitorResize : true,
  301.     triggerWidth : 18,
  302.     lastOverflow : false,
  303.     noItemsMenuText : '<div class="x-toolbar-no-items">(None)</div>',
  304.     // private
  305.     onLayout : function(ct, target){
  306.         if(!this.leftTr){
  307.             target.addClass('x-toolbar-layout-ct');
  308.             target.insertHtml('beforeEnd',
  309.                  '<table cellspacing="0" class="x-toolbar-ct"><tbody><tr><td class="x-toolbar-left" align="left"><table cellspacing="0"><tbody><tr class="x-toolbar-left-row"></tr></tbody></table></td><td class="x-toolbar-right" align="right"><table cellspacing="0" class="x-toolbar-right-ct"><tbody><tr><td><table cellspacing="0"><tbody><tr class="x-toolbar-right-row"></tr></tbody></table></td><td><table cellspacing="0"><tbody><tr class="x-toolbar-extras-row"></tr></tbody></table></td></tr></tbody></table></td></tr></tbody></table>');
  310.             this.leftTr = target.child('tr.x-toolbar-left-row', true);
  311.             this.rightTr = target.child('tr.x-toolbar-right-row', true);
  312.             this.extrasTr = target.child('tr.x-toolbar-extras-row', true);
  313.         }
  314.         var side = this.leftTr;
  315.         var pos = 0;
  316.         var items = ct.items.items;
  317.         for(var i = 0, len = items.length, c; i < len; i++, pos++) {
  318.             c = items[i];
  319.             if(c.isFill){
  320.                 side = this.rightTr;
  321.                 pos = -1;
  322.             }else if(!c.rendered){
  323.                 c.render(this.insertCell(c, side, pos));
  324.             }else{
  325.                 if(!c.xtbHidden && !this.isValidParent(c, side.childNodes[pos])){
  326.                     var td = this.insertCell(c, side, pos);
  327.                     td.appendChild(c.getDomPositionEl().dom);
  328.                     c.container = Ext.get(td);
  329.                 }
  330.             }
  331.         }
  332.         //strip extra empty cells
  333.         this.cleanup(this.leftTr);
  334.         this.cleanup(this.rightTr);
  335.         this.cleanup(this.extrasTr);
  336.         this.fitToSize(target);
  337.     },
  338.     cleanup : function(row){
  339.         var cn = row.childNodes;
  340.         for(var i = cn.length-1, c; i >= 0 && (c = cn[i]); i--){
  341.             if(!c.firstChild){
  342.                 row.removeChild(c);
  343.             }
  344.         }
  345.     },
  346.     insertCell : function(c, side, pos){
  347.         var td = document.createElement('td');
  348.         td.className='x-toolbar-cell';
  349.         side.insertBefore(td, side.childNodes[pos]||null);
  350.         return td;
  351.     },
  352.     hideItem : function(item){
  353.         var h = (this.hiddens = this.hiddens || []);
  354.         h.push(item);
  355.         item.xtbHidden = true;
  356.         item.xtbWidth = item.getDomPositionEl().dom.parentNode.offsetWidth;
  357.         item.hide();
  358.     },
  359.     unhideItem : function(item){
  360.         item.show();
  361.         item.xtbHidden = false;
  362.         this.hiddens.remove(item);
  363.         if(this.hiddens.length < 1){
  364.             delete this.hiddens;
  365.         }
  366.     },
  367.     getItemWidth : function(c){
  368.         return c.hidden ? (c.xtbWidth || 0) : c.getDomPositionEl().dom.parentNode.offsetWidth;
  369.     },
  370.     fitToSize : function(t){
  371.         if(this.container.enableOverflow === false){
  372.             return;
  373.         }
  374.         var w = t.dom.clientWidth;
  375.         var lw = this.lastWidth || 0;
  376.         this.lastWidth = w;
  377.         var iw = t.dom.firstChild.offsetWidth;
  378.         var clipWidth = w - this.triggerWidth;
  379.         var hideIndex = -1;
  380.         if(iw > w || (this.hiddens && w >= lw)){
  381.             var i, items = this.container.items.items, len = items.length, c;
  382.             var loopWidth = 0;
  383.             for(i = 0; i < len; i++) {
  384.                 c = items[i];
  385.                 if(!c.isFill){
  386.                     loopWidth += this.getItemWidth(c);
  387.                     if(loopWidth > clipWidth){
  388.                         if(!c.xtbHidden){
  389.                             this.hideItem(c);
  390.                         }
  391.                     }else{
  392.                         if(c.xtbHidden){
  393.                             this.unhideItem(c);
  394.                         }
  395.                     }
  396.                 }
  397.             }
  398.         }
  399.         if(this.hiddens){
  400.             this.initMore();
  401.             if(!this.lastOverflow){
  402.                 this.container.fireEvent('overflowchange', this.container, true);
  403.                 this.lastOverflow = true;
  404.             }
  405.         }else if(this.more){
  406.             this.clearMenu();
  407.             this.more.destroy();
  408.             delete this.more;
  409.             if(this.lastOverflow){
  410.                 this.container.fireEvent('overflowchange', this.container, false);
  411.                 this.lastOverflow = false;
  412.             }
  413.         }
  414.     },
  415.     createMenuConfig : function(c, hideOnClick){
  416.         var cfg = Ext.apply({}, c.initialConfig),
  417.             group = c.toggleGroup;
  418.         Ext.apply(cfg, {
  419.             text: c.overflowText || c.text,
  420.             iconCls: c.iconCls,
  421.             icon: c.icon,
  422.             itemId: c.itemId,
  423.             disabled: c.disabled,
  424.             handler: c.handler,
  425.             scope: c.scope,
  426.             menu: c.menu,
  427.             hideOnClick: hideOnClick
  428.         });
  429.         if(group || c.enableToggle){
  430.             Ext.apply(cfg, {
  431.                 group: group,
  432.                 checked: c.pressed,
  433.                 listeners: {
  434.                     checkchange: function(item, checked){
  435.                         c.toggle(checked);
  436.                     }
  437.                 }
  438.             });
  439.         }
  440.         delete cfg.xtype;
  441.         delete cfg.id;
  442.         return cfg;
  443.     },
  444.     // private
  445.     addComponentToMenu : function(m, c){
  446.         if(c instanceof Ext.Toolbar.Separator){
  447.             m.add('-');
  448.         }else if(Ext.isFunction(c.isXType)){
  449.             if(c.isXType('splitbutton')){
  450.                 m.add(this.createMenuConfig(c, true));
  451.             }else if(c.isXType('button')){
  452.                 m.add(this.createMenuConfig(c, !c.menu));
  453.             }else if(c.isXType('buttongroup')){
  454.                 c.items.each(function(item){
  455.                      this.addComponentToMenu(m, item);
  456.                 }, this);
  457.             }
  458.         }
  459.     },
  460.     clearMenu : function(){
  461.         var m = this.moreMenu;
  462.         if(m && m.items){
  463.             this.moreMenu.items.each(function(item){
  464.                 delete item.menu;
  465.             });
  466.         }
  467.     },
  468.     // private
  469.     beforeMoreShow : function(m){
  470.         var h = this.container.items.items,
  471.             len = h.length,
  472.             c,
  473.             prev,
  474.             needsSep = function(group, item){
  475.                 return group.isXType('buttongroup') && !(item instanceof Ext.Toolbar.Separator);
  476.             };
  477.         this.clearMenu();
  478.         m.removeAll();
  479.         for(var i = 0; i < len; i++){
  480.             c = h[i];
  481.             if(c.xtbHidden){
  482.                 if(prev && (needsSep(c, prev) || needsSep(prev, c))){
  483.                     m.add('-');
  484.                 }
  485.                 this.addComponentToMenu(m, c);
  486.                 prev = c;
  487.             }
  488.         }
  489.         // put something so the menu isn't empty
  490.         // if no compatible items found
  491.         if(m.items.length < 1){
  492.             m.add(this.noItemsMenuText);
  493.         }
  494.     },
  495.     initMore : function(){
  496.         if(!this.more){
  497.             this.moreMenu = new Ext.menu.Menu({
  498.                 listeners: {
  499.                     beforeshow: this.beforeMoreShow,
  500.                     scope: this
  501.                 }
  502.             });
  503.             this.more = new Ext.Button({
  504.                 iconCls: 'x-toolbar-more-icon',
  505.                 cls: 'x-toolbar-more',
  506.                 menu: this.moreMenu
  507.             });
  508.             var td = this.insertCell(this.more, this.extrasTr, 100);
  509.             this.more.render(td);
  510.         }
  511.     },
  512.     destroy : function(){
  513.         Ext.destroy(this.more, this.moreMenu);
  514.         Ext.layout.ToolbarLayout.superclass.destroy.call(this);
  515.     }
  516.     /**
  517.      * @property activeItem
  518.      * @hide
  519.      */
  520. });
  521. Ext.Container.LAYOUTS.toolbar = Ext.layout.ToolbarLayout;
  522. /**
  523.  * @class Ext.Toolbar
  524.  * @extends Ext.Container
  525.  * <p>Basic Toolbar class. Although the <tt>{@link Ext.Container#defaultType defaultType}</tt> for Toolbar
  526.  * is <tt>{@link Ext.Button button}</tt>, Toolbar elements (child items for the Toolbar container) may
  527.  * be virtually any type of Component. Toolbar elements can be created explicitly via their constructors,
  528.  * or implicitly via their xtypes, and can be <tt>{@link #add}</tt>ed dynamically.</p>
  529.  * <p>Some items have shortcut strings for creation:</p>
  530.  * <pre>
  531. <u>Shortcut</u>  <u>xtype</u>          <u>Class</u>                  <u>Description</u>
  532. '->'      'tbfill'       {@link Ext.Toolbar.Fill}       begin using the right-justified button container
  533. '-'       'tbseparator'  {@link Ext.Toolbar.Separator}  add a vertical separator bar between toolbar items
  534. ' '       'tbspacer'     {@link Ext.Toolbar.Spacer}     add horiztonal space between elements
  535.  * </pre>
  536.  *
  537.  * Example usage of various elements:
  538.  * <pre><code>
  539. var tb = new Ext.Toolbar({
  540.     renderTo: document.body,
  541.     width: 600,
  542.     height: 100,
  543.     items: [
  544.         {
  545.             // xtype: 'button', // default for Toolbars, same as 'tbbutton'
  546.             text: 'Button'
  547.         },
  548.         {
  549.             xtype: 'splitbutton', // same as 'tbsplitbutton'
  550.             text: 'Split Button'
  551.         },
  552.         // begin using the right-justified button container
  553.         '->', // same as {xtype: 'tbfill'}, // Ext.Toolbar.Fill
  554.         {
  555.             xtype: 'textfield',
  556.             name: 'field1',
  557.             emptyText: 'enter search term'
  558.         },
  559.         // add a vertical separator bar between toolbar items
  560.         '-', // same as {xtype: 'tbseparator'} to create Ext.Toolbar.Separator
  561.         'text 1', // same as {xtype: 'tbtext', text: 'text1'} to create Ext.Toolbar.TextItem
  562.         {xtype: 'tbspacer'},// same as ' ' to create Ext.Toolbar.Spacer
  563.         'text 2',
  564.         {xtype: 'tbspacer', width: 50}, // add a 50px space
  565.         'text 3'
  566.     ]
  567. });
  568.  * </code></pre>
  569.  * Example adding a ComboBox within a menu of a button:
  570.  * <pre><code>
  571. // ComboBox creation
  572. var combo = new Ext.form.ComboBox({
  573.     store: new Ext.data.ArrayStore({
  574.         autoDestroy: true,
  575.         fields: ['initials', 'fullname'],
  576.         data : [
  577.             ['FF', 'Fred Flintstone'],
  578.             ['BR', 'Barney Rubble']
  579.         ]
  580.     }),
  581.     displayField: 'fullname',
  582.     typeAhead: true,
  583.     mode: 'local',
  584.     forceSelection: true,
  585.     triggerAction: 'all',
  586.     emptyText: 'Select a name...',
  587.     selectOnFocus: true,
  588.     width: 135,
  589.     getListParent: function() {
  590.         return this.el.up('.x-menu');
  591.     },
  592.     iconCls: 'no-icon' //use iconCls if placing within menu to shift to right side of menu
  593. });
  594. // put ComboBox in a Menu
  595. var menu = new Ext.menu.Menu({
  596.     id: 'mainMenu',
  597.     items: [
  598.         combo // A Field in a Menu
  599.     ]
  600. });
  601. // add a Button with the menu
  602. tb.add({
  603.         text:'Button w/ Menu',
  604.         menu: menu  // assign menu by instance
  605.     });
  606. tb.doLayout();
  607.  * </code></pre>
  608.  * @constructor
  609.  * Creates a new Toolbar
  610.  * @param {Object/Array} config A config object or an array of buttons to <tt>{@link #add}</tt>
  611.  * @xtype toolbar
  612.  */
  613. Ext.Toolbar = function(config){
  614.     if(Ext.isArray(config)){
  615.         config = {items: config, layout: 'toolbar'};
  616.     } else {
  617.         config = Ext.apply({
  618.             layout: 'toolbar'
  619.         }, config);
  620.         if(config.buttons) {
  621.             config.items = config.buttons;
  622.         }
  623.     }
  624.     Ext.Toolbar.superclass.constructor.call(this, config);
  625. };
  626. (function(){
  627. var T = Ext.Toolbar;
  628. Ext.extend(T, Ext.Container, {
  629.     defaultType: 'button',
  630.     trackMenus : true,
  631.     internalDefaults: {removeMode: 'container', hideParent: true},
  632.     toolbarCls: 'x-toolbar',
  633.     initComponent : function(){
  634.         T.superclass.initComponent.call(this);
  635.         /**
  636.          * @event overflowchange
  637.          * Fires after the overflow state has changed.
  638.          * @param {Object} c The Container
  639.          * @param {Boolean} lastOverflow overflow state
  640.          */
  641.         this.addEvents('overflowchange');
  642.     },
  643.     // private
  644.     onRender : function(ct, position){
  645.         if(!this.el){
  646.             if(!this.autoCreate){
  647.                 this.autoCreate = {
  648.                     cls: this.toolbarCls + ' x-small-editor'
  649.                 };
  650.             }
  651.             this.el = ct.createChild(Ext.apply({ id: this.id },this.autoCreate), position);
  652.         }
  653.     },
  654.     /**
  655.      * Adds element(s) to the toolbar -- this function takes a variable number of
  656.      * arguments of mixed type and adds them to the toolbar.
  657.      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
  658.      * <ul>
  659.      * <li>{@link Ext.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
  660.      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
  661.      * <li>Field: Any form field (equivalent to {@link #addField})</li>
  662.      * <li>Item: Any subclass of {@link Ext.Toolbar.Item} (equivalent to {@link #addItem})</li>
  663.      * <li>String: Any generic string (gets wrapped in a {@link Ext.Toolbar.TextItem}, equivalent to {@link #addText}).
  664.      * Note that there are a few special strings that are treated differently as explained next.</li>
  665.      * <li>'-': Creates a separator element (equivalent to {@link #addSeparator})</li>
  666.      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
  667.      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
  668.      * </ul>
  669.      * @param {Mixed} arg2
  670.      * @param {Mixed} etc.
  671.      * @method add
  672.      */
  673.     // private
  674.     lookupComponent : function(c){
  675.         if(Ext.isString(c)){
  676.             if(c == '-'){
  677.                 c = new T.Separator();
  678.             }else if(c == ' '){
  679.                 c = new T.Spacer();
  680.             }else if(c == '->'){
  681.                 c = new T.Fill();
  682.             }else{
  683.                 c = new T.TextItem(c);
  684.             }
  685.             this.applyDefaults(c);
  686.         }else{
  687.             if(c.isFormField || c.render){ // some kind of form field, some kind of Toolbar.Item
  688.                 c = this.constructItem(c);
  689.             }else if(c.tag){ // DomHelper spec
  690.                 c = new T.Item({autoEl: c});
  691.             }else if(c.tagName){ // element
  692.                 c = new T.Item({el:c});
  693.             }else if(Ext.isObject(c)){ // must be button config?
  694.                 c = c.xtype ? this.constructItem(c) : this.constructButton(c);
  695.             }
  696.         }
  697.         return c;
  698.     },
  699.     // private
  700.     applyDefaults : function(c){
  701.         if(!Ext.isString(c)){
  702.             c = Ext.Toolbar.superclass.applyDefaults.call(this, c);
  703.             var d = this.internalDefaults;
  704.             if(c.events){
  705.                 Ext.applyIf(c.initialConfig, d);
  706.                 Ext.apply(c, d);
  707.             }else{
  708.                 Ext.applyIf(c, d);
  709.             }
  710.         }
  711.         return c;
  712.     },
  713.     // private
  714.     constructItem : function(item, type){
  715.         return Ext.create(item, type || this.defaultType);
  716.     },
  717.     /**
  718.      * Adds a separator
  719.      * @return {Ext.Toolbar.Item} The separator {@link Ext.Toolbar.Item item}
  720.      */
  721.     addSeparator : function(){
  722.         return this.add(new T.Separator());
  723.     },
  724.     /**
  725.      * Adds a spacer element
  726.      * @return {Ext.Toolbar.Spacer} The spacer item
  727.      */
  728.     addSpacer : function(){
  729.         return this.add(new T.Spacer());
  730.     },
  731.     /**
  732.      * Forces subsequent additions into the float:right toolbar
  733.      */
  734.     addFill : function(){
  735.         this.add(new T.Fill());
  736.     },
  737.     /**
  738.      * Adds any standard HTML element to the toolbar
  739.      * @param {Mixed} el The element or id of the element to add
  740.      * @return {Ext.Toolbar.Item} The element's item
  741.      */
  742.     addElement : function(el){
  743.         return this.addItem(new T.Item({el:el}));
  744.     },
  745.     /**
  746.      * Adds any Toolbar.Item or subclass
  747.      * @param {Ext.Toolbar.Item} item
  748.      * @return {Ext.Toolbar.Item} The item
  749.      */
  750.     addItem : function(item){
  751.         return Ext.Toolbar.superclass.add.apply(this, arguments);
  752.     },
  753.     /**
  754.      * Adds a button (or buttons). See {@link Ext.Button} for more info on the config.
  755.      * @param {Object/Array} config A button config or array of configs
  756.      * @return {Ext.Button/Array}
  757.      */
  758.     addButton : function(config){
  759.         if(Ext.isArray(config)){
  760.             var buttons = [];
  761.             for(var i = 0, len = config.length; i < len; i++) {
  762.                 buttons.push(this.addButton(config[i]));
  763.             }
  764.             return buttons;
  765.         }
  766.         return this.add(this.constructButton(config));
  767.     },
  768.     /**
  769.      * Adds text to the toolbar
  770.      * @param {String} text The text to add
  771.      * @return {Ext.Toolbar.Item} The element's item
  772.      */
  773.     addText : function(text){
  774.         return this.addItem(new T.TextItem(text));
  775.     },
  776.     /**
  777.      * Adds a new element to the toolbar from the passed {@link Ext.DomHelper} config
  778.      * @param {Object} config
  779.      * @return {Ext.Toolbar.Item} The element's item
  780.      */
  781.     addDom : function(config){
  782.         return this.add(new T.Item({autoEl: config}));
  783.     },
  784.     /**
  785.      * Adds a dynamically rendered Ext.form field (TextField, ComboBox, etc). Note: the field should not have
  786.      * been rendered yet. For a field that has already been rendered, use {@link #addElement}.
  787.      * @param {Ext.form.Field} field
  788.      * @return {Ext.Toolbar.Item}
  789.      */
  790.     addField : function(field){
  791.         return this.add(field);
  792.     },
  793.     /**
  794.      * Inserts any {@link Ext.Toolbar.Item}/{@link Ext.Button} at the specified index.
  795.      * @param {Number} index The index where the item is to be inserted
  796.      * @param {Object/Ext.Toolbar.Item/Ext.Button/Array} item The button, or button config object to be
  797.      * inserted, or an array of buttons/configs.
  798.      * @return {Ext.Button/Item}
  799.      */
  800.     insertButton : function(index, item){
  801.         if(Ext.isArray(item)){
  802.             var buttons = [];
  803.             for(var i = 0, len = item.length; i < len; i++) {
  804.                buttons.push(this.insertButton(index + i, item[i]));
  805.             }
  806.             return buttons;
  807.         }
  808.         return Ext.Toolbar.superclass.insert.call(this, index, item);
  809.     },
  810.     // private
  811.     initMenuTracking : function(item){
  812.         if(this.trackMenus && item.menu){
  813.             this.mon(item, {
  814.                 'menutriggerover' : this.onButtonTriggerOver,
  815.                 'menushow' : this.onButtonMenuShow,
  816.                 'menuhide' : this.onButtonMenuHide,
  817.                 scope: this
  818.             });
  819.         }
  820.     },
  821.     // private
  822.     constructButton : function(item){
  823.         var b = item.events ? item : this.constructItem(item, item.split ? 'splitbutton' : this.defaultType);
  824.         this.initMenuTracking(b);
  825.         return b;
  826.     },
  827.     // private
  828.     onDisable : function(){
  829.         this.items.each(function(item){
  830.              if(item.disable){
  831.                  item.disable();
  832.              }
  833.         });
  834.     },
  835.     // private
  836.     onEnable : function(){
  837.         this.items.each(function(item){
  838.              if(item.enable){
  839.                  item.enable();
  840.              }
  841.         });
  842.     },
  843.     // private
  844.     onButtonTriggerOver : function(btn){
  845.         if(this.activeMenuBtn && this.activeMenuBtn != btn){
  846.             this.activeMenuBtn.hideMenu();
  847.             btn.showMenu();
  848.             this.activeMenuBtn = btn;
  849.         }
  850.     },
  851.     // private
  852.     onButtonMenuShow : function(btn){
  853.         this.activeMenuBtn = btn;
  854.     },
  855.     // private
  856.     onButtonMenuHide : function(btn){
  857.         delete this.activeMenuBtn;
  858.     }
  859. });
  860. Ext.reg('toolbar', Ext.Toolbar);
  861. /**
  862.  * @class Ext.Toolbar.Item
  863.  * @extends Ext.BoxComponent
  864.  * The base class that other non-interacting Toolbar Item classes should extend in order to
  865.  * get some basic common toolbar item functionality.
  866.  * @constructor
  867.  * Creates a new Item
  868.  * @param {HTMLElement} el
  869.  * @xtype tbitem
  870.  */
  871. T.Item = Ext.extend(Ext.BoxComponent, {
  872.     hideParent: true, //  Hiding a Toolbar.Item hides its containing TD
  873.     enable:Ext.emptyFn,
  874.     disable:Ext.emptyFn,
  875.     focus:Ext.emptyFn
  876.     /**
  877.      * @cfg {String} overflowText Text to be used for the menu if the item is overflowed.
  878.      */
  879. });
  880. Ext.reg('tbitem', T.Item);
  881. /**
  882.  * @class Ext.Toolbar.Separator
  883.  * @extends Ext.Toolbar.Item
  884.  * A simple class that adds a vertical separator bar between toolbar items
  885.  * (css class:<tt>'xtb-sep'</tt>). Example usage:
  886.  * <pre><code>
  887. new Ext.Panel({
  888.     tbar : [
  889.         'Item 1',
  890.         {xtype: 'tbseparator'}, // or '-'
  891.         'Item 2'
  892.     ]
  893. });
  894. </code></pre>
  895.  * @constructor
  896.  * Creates a new Separator
  897.  * @xtype tbseparator
  898.  */
  899. T.Separator = Ext.extend(T.Item, {
  900.     onRender : function(ct, position){
  901.         this.el = ct.createChild({tag:'span', cls:'xtb-sep'}, position);
  902.     }
  903. });
  904. Ext.reg('tbseparator', T.Separator);
  905. /**
  906.  * @class Ext.Toolbar.Spacer
  907.  * @extends Ext.Toolbar.Item
  908.  * A simple element that adds extra horizontal space between items in a toolbar.
  909.  * By default a 2px wide space is added via css specification:<pre><code>
  910. .x-toolbar .xtb-spacer {
  911.     width:2px;
  912. }
  913.  * </code></pre>
  914.  * <p>Example usage:</p>
  915.  * <pre><code>
  916. new Ext.Panel({
  917.     tbar : [
  918.         'Item 1',
  919.         {xtype: 'tbspacer'}, // or ' '
  920.         'Item 2',
  921.         // space width is also configurable via javascript
  922.         {xtype: 'tbspacer', width: 50}, // add a 50px space
  923.         'Item 3'
  924.     ]
  925. });
  926. </code></pre>
  927.  * @constructor
  928.  * Creates a new Spacer
  929.  * @xtype tbspacer
  930.  */
  931. T.Spacer = Ext.extend(T.Item, {
  932.     /**
  933.      * @cfg {Number} width
  934.      * The width of the spacer in pixels (defaults to 2px via css style <tt>.x-toolbar .xtb-spacer</tt>).
  935.      */
  936.     onRender : function(ct, position){
  937.         this.el = ct.createChild({tag:'div', cls:'xtb-spacer', style: this.width?'width:'+this.width+'px':''}, position);
  938.     }
  939. });
  940. Ext.reg('tbspacer', T.Spacer);
  941. /**
  942.  * @class Ext.Toolbar.Fill
  943.  * @extends Ext.Toolbar.Spacer
  944.  * A non-rendering placeholder item which instructs the Toolbar's Layout to begin using
  945.  * the right-justified button container.
  946.  * <pre><code>
  947. new Ext.Panel({
  948.     tbar : [
  949.         'Item 1',
  950.         {xtype: 'tbfill'}, // or '->'
  951.         'Item 2'
  952.     ]
  953. });
  954. </code></pre>
  955.  * @constructor
  956.  * Creates a new Fill
  957.  * @xtype tbfill
  958.  */
  959. T.Fill = Ext.extend(T.Item, {
  960.     // private
  961.     render : Ext.emptyFn,
  962.     isFill : true
  963. });
  964. Ext.reg('tbfill', T.Fill);
  965. /**
  966.  * @class Ext.Toolbar.TextItem
  967.  * @extends Ext.Toolbar.Item
  968.  * A simple class that renders text directly into a toolbar
  969.  * (with css class:<tt>'xtb-text'</tt>). Example usage:
  970.  * <pre><code>
  971. new Ext.Panel({
  972.     tbar : [
  973.         {xtype: 'tbtext', text: 'Item 1'} // or simply 'Item 1'
  974.     ]
  975. });
  976. </code></pre>
  977.  * @constructor
  978.  * Creates a new TextItem
  979.  * @param {String/Object} text A text string, or a config object containing a <tt>text</tt> property
  980.  * @xtype tbtext
  981.  */
  982. T.TextItem = Ext.extend(T.Item, {
  983.     constructor: function(config){
  984.         if (Ext.isString(config)) {
  985.             config = { autoEl: {cls: 'xtb-text', html: config }};
  986.         } else {
  987.             config.autoEl = {cls: 'xtb-text', html: config.text || ''};
  988.         }
  989.         T.TextItem.superclass.constructor.call(this, config);
  990.     },
  991.     /**
  992.      * Updates this item's text, setting the text to be used as innerHTML.
  993.      * @param {String} t The text to display (html accepted).
  994.      */
  995.     setText : function(t) {
  996.         if (this.rendered) {
  997.             this.el.dom.innerHTML = t;
  998.         } else {
  999.             this.autoEl.html = t;
  1000.         }
  1001.     }
  1002. });
  1003. Ext.reg('tbtext', T.TextItem);
  1004. // backwards compat
  1005. T.Button = Ext.extend(Ext.Button, {});
  1006. T.SplitButton = Ext.extend(Ext.SplitButton, {});
  1007. Ext.reg('tbbutton', T.Button);
  1008. Ext.reg('tbsplit', T.SplitButton);
  1009. })();
  1010. /**
  1011.  * @class Ext.ButtonGroup
  1012.  * @extends Ext.Panel
  1013.  * Container for a group of buttons. Example usage:
  1014.  * <pre><code>
  1015. var p = new Ext.Panel({
  1016.     title: 'Panel with Button Group',
  1017.     width: 300,
  1018.     height:200,
  1019.     renderTo: document.body,
  1020.     html: 'whatever',
  1021.     tbar: [{
  1022.         xtype: 'buttongroup',
  1023.         {@link #columns}: 3,
  1024.         title: 'Clipboard',
  1025.         items: [{
  1026.             text: 'Paste',
  1027.             scale: 'large',
  1028.             rowspan: 3, iconCls: 'add',
  1029.             iconAlign: 'top',
  1030.             cls: 'x-btn-as-arrow'
  1031.         },{
  1032.             xtype:'splitbutton',
  1033.             text: 'Menu Button',
  1034.             scale: 'large',
  1035.             rowspan: 3,
  1036.             iconCls: 'add',
  1037.             iconAlign: 'top',
  1038.             arrowAlign:'bottom',
  1039.             menu: [{text: 'Menu Item 1'}]
  1040.         },{
  1041.             xtype:'splitbutton', text: 'Cut', iconCls: 'add16', menu: [{text: 'Cut Menu Item'}]
  1042.         },{
  1043.             text: 'Copy', iconCls: 'add16'
  1044.         },{
  1045.             text: 'Format', iconCls: 'add16'
  1046.         }]
  1047.     }]
  1048. });
  1049.  * </code></pre>
  1050.  * @xtype buttongroup
  1051.  */
  1052. Ext.ButtonGroup = Ext.extend(Ext.Panel, {
  1053.     /**
  1054.      * @cfg {Number} columns The <tt>columns</tt> configuration property passed to the
  1055.      * {@link #layout configured layout manager}. See {@link Ext.layout.TableLayout#columns}.
  1056.      */
  1057.     /**
  1058.      * @cfg {String} baseCls  Defaults to <tt>'x-btn-group'</tt>.  See {@link Ext.Panel#baseCls}.
  1059.      */
  1060.     baseCls: 'x-btn-group',
  1061.     /**
  1062.      * @cfg {String} layout  Defaults to <tt>'table'</tt>.  See {@link Ext.Container#layout}.
  1063.      */
  1064.     layout:'table',
  1065.     defaultType: 'button',
  1066.     /**
  1067.      * @cfg {Boolean} frame  Defaults to <tt>true</tt>.  See {@link Ext.Panel#frame}.
  1068.      */
  1069.     frame: true,
  1070.     internalDefaults: {removeMode: 'container', hideParent: true},
  1071.     initComponent : function(){
  1072.         this.layoutConfig = this.layoutConfig || {};
  1073.         Ext.applyIf(this.layoutConfig, {
  1074.             columns : this.columns
  1075.         });
  1076.         if(!this.title){
  1077.             this.addClass('x-btn-group-notitle');
  1078.         }
  1079.         this.on('afterlayout', this.onAfterLayout, this);
  1080.         Ext.ButtonGroup.superclass.initComponent.call(this);
  1081.     },
  1082.     applyDefaults : function(c){
  1083.         c = Ext.ButtonGroup.superclass.applyDefaults.call(this, c);
  1084.         var d = this.internalDefaults;
  1085.         if(c.events){
  1086.             Ext.applyIf(c.initialConfig, d);
  1087.             Ext.apply(c, d);
  1088.         }else{
  1089.             Ext.applyIf(c, d);
  1090.         }
  1091.         return c;
  1092.     },
  1093.     onAfterLayout : function(){
  1094.         var bodyWidth = this.body.getFrameWidth('lr') + this.body.dom.firstChild.offsetWidth;
  1095.         this.body.setWidth(bodyWidth);
  1096.         this.el.setWidth(bodyWidth + this.getFrameWidth());
  1097.     }
  1098.     /**
  1099.      * @cfg {Array} tools  @hide
  1100.      */
  1101. });
  1102. Ext.reg('buttongroup', Ext.ButtonGroup);
  1103. /**  * @class Ext.PagingToolbar  * @extends Ext.Toolbar  * <p>As the amount of records increases, the time required for the browser to render  * them increases. Paging is used to reduce the amount of data exchanged with the client.  * Note: if there are more records/rows than can be viewed in the available screen area, vertical  * scrollbars will be added.</p>  * <p>Paging is typically handled on the server side (see exception below). The client sends  * parameters to the server side, which the server needs to interpret and then respond with the  * approprate data.</p>  * <p><b>Ext.PagingToolbar</b> is a specialized toolbar that is bound to a {@link Ext.data.Store}  * and provides automatic paging control. This Component {@link Ext.data.Store#load load}s blocks  * of data into the <tt>{@link #store}</tt> by passing {@link Ext.data.Store#paramNames paramNames} used for  * paging criteria.</p>  * <p>PagingToolbar is typically used as one of the Grid's toolbars:</p>  * <pre><code> Ext.QuickTips.init(); // to display button quicktips var myStore = new Ext.data.Store({     ... }); var myPageSize = 25;  // server script should only send back 25 items var grid = new Ext.grid.GridPanel({     ...     store: myStore,     bbar: new Ext.PagingToolbar({         {@link #store}: myStore,       // grid and PagingToolbar using same store         {@link #displayInfo}: true,         {@link #pageSize}: myPageSize,         {@link #prependButtons}: true,         items: [             'text 1'         ]     }) });  * </code></pre>  *  * <p>To use paging, pass the paging requirements to the server when the store is first loaded.</p>  * <pre><code> store.load({     params: {         start: 0,          // specify params for the first page load if using paging         limit: myPageSize,         foo:   'bar'     } });  * </code></pre>  * <p><u>Paging with Local Data</u></p>  * <p>Paging can also be accomplished with local data using extensions:</p>  * <div class="mdetail-params"><ul>  * <li><a href="http://extjs.com/forum/showthread.php?t=57386">Ext.ux.data.PagingStore</a></li>  * <li>Paging Memory Proxy (examples/ux/PagingMemoryProxy.js)</li>  * </ul></div>  * @constructor  * Create a new PagingToolbar  * @param {Object} config The config object  * @xtype paging  */ (function() { var T = Ext.Toolbar; Ext.PagingToolbar = Ext.extend(Ext.Toolbar, {     /**      * @cfg {Ext.data.Store} store      * The {@link Ext.data.Store} the paging toolbar should use as its data source (required).      */     /**      * @cfg {Boolean} displayInfo      * <tt>true</tt> to display the displayMsg (defaults to <tt>false</tt>)      */     /**      * @cfg {Number} pageSize      * The number of records to display per page (defaults to <tt>20</tt>)      */     pageSize : 20,     /**      * @cfg {Boolean} prependButtons      * <tt>true</tt> to insert any configured <tt>items</tt> <i>before</i> the paging buttons.      * Defaults to <tt>false</tt>.      */     /**      * @cfg {String} displayMsg      * The paging status message to display (defaults to <tt>'Displaying {0} - {1} of {2}'</tt>).      * Note that this string is formatted using the braced numbers <tt>{0}-{2}</tt> as tokens      * that are replaced by the values for start, end and total respectively. These tokens should      * be preserved when overriding this string if showing those values is desired.      */     displayMsg : 'Displaying {0} - {1} of {2}',     /**      * @cfg {String} emptyMsg      * The message to display when no records are found (defaults to 'No data to display')      */     emptyMsg : 'No data to display',     /**      * @cfg {String} beforePageText      * The text displayed before the input item (defaults to <tt>'Page'</tt>).      */     beforePageText : 'Page',     /**      * @cfg {String} afterPageText      * Customizable piece of the default paging text (defaults to <tt>'of {0}'</tt>). Note that      * this string is formatted using <tt>{0}</tt> as a token that is replaced by the number of      * total pages. This token should be preserved when overriding this string if showing the      * total page count is desired.      */     afterPageText : 'of {0}',     /**      * @cfg {String} firstText      * The quicktip text displayed for the first page button (defaults to <tt>'First Page'</tt>).      * <b>Note</b>: quick tips must be initialized for the quicktip to show.      */     firstText : 'First Page',     /**      * @cfg {String} prevText      * The quicktip text displayed for the previous page button (defaults to <tt>'Previous Page'</tt>).      * <b>Note</b>: quick tips must be initialized for the quicktip to show.      */     prevText : 'Previous Page',     /**      * @cfg {String} nextText      * The quicktip text displayed for the next page button (defaults to <tt>'Next Page'</tt>).      * <b>Note</b>: quick tips must be initialized for the quicktip to show.      */     nextText : 'Next Page',     /**      * @cfg {String} lastText      * The quicktip text displayed for the last page button (defaults to <tt>'Last Page'</tt>).      * <b>Note</b>: quick tips must be initialized for the quicktip to show.      */     lastText : 'Last Page',     /**      * @cfg {String} refreshText      * The quicktip text displayed for the Refresh button (defaults to <tt>'Refresh'</tt>).      * <b>Note</b>: quick tips must be initialized for the quicktip to show.      */     refreshText : 'Refresh',     /**      * @deprecated      * <b>The defaults for these should be set in the data store.</b>      * Object mapping of parameter names used for load calls, initially set to:      * <pre>{start: 'start', limit: 'limit'}</pre>      */     /**      * The number of records to display per page.  See also <tt>{@link #cursor}</tt>.      * @type Number      * @property pageSize      */     /**      * Indicator for the record position.  This property might be used to get the active page      * number for example:<pre><code>      * // t is reference to the paging toolbar instance      * var activePage = Math.ceil((t.cursor + t.pageSize) / t.pageSize);      * </code></pre>      * @type Number      * @property cursor      */     initComponent : function(){         var pagingItems = [this.first = new T.Button({             tooltip: this.firstText,             overflowText: this.firstText,             iconCls: 'x-tbar-page-first',             disabled: true,             handler: this.moveFirst,             scope: this         }), this.prev = new T.Button({             tooltip: this.prevText,             overflowText: this.prevText,             iconCls: 'x-tbar-page-prev',             disabled: true,             handler: this.movePrevious,             scope: this         }), '-', this.beforePageText,         this.inputItem = new Ext.form.NumberField({             cls: 'x-tbar-page-number',             allowDecimals: false,             allowNegative: false,             enableKeyEvents: true,             selectOnFocus: true,             listeners: {                 scope: this,                 keydown: this.onPagingKeyDown,                 blur: this.onPagingBlur             }         }), this.afterTextItem = new T.TextItem({             text: String.format(this.afterPageText, 1)         }), '-', this.next = new T.Button({             tooltip: this.nextText,             overflowText: this.nextText,             iconCls: 'x-tbar-page-next',             disabled: true,             handler: this.moveNext,             scope: this         }), this.last = new T.Button({             tooltip: this.lastText,             overflowText: this.lastText,             iconCls: 'x-tbar-page-last',             disabled: true,             handler: this.moveLast,             scope: this         }), '-', this.refresh = new T.Button({             tooltip: this.refreshText,             overflowText: this.refreshText,             iconCls: 'x-tbar-loading',             handler: this.refresh,             scope: this         })];         var userItems = this.items || this.buttons || [];         if (this.prependButtons) {             this.items = userItems.concat(pagingItems);         }else{             this.items = pagingItems.concat(userItems);         }         delete this.buttons;         if(this.displayInfo){             this.items.push('->');             this.items.push(this.displayItem = new T.TextItem({}));         }         Ext.PagingToolbar.superclass.initComponent.call(this);         this.addEvents(             /**              * @event change              * Fires after the active page has been changed.              * @param {Ext.PagingToolbar} this              * @param {Object} pageData An object that has these properties:<ul>              * <li><code>total</code> : Number <div class="sub-desc">The total number of records in the dataset as              * returned by the server</div></li>              * <li><code>activePage</code> : Number <div class="sub-desc">The current page number</div></li>              * <li><code>pages</code> : Number <div class="sub-desc">The total number of pages (calculated from              * the total number of records in the dataset as returned by the server and the current {@link #pageSize})</div></li>              * </ul>              */             'change',             /**              * @event beforechange              * Fires just before the active page is changed.              * Return false to prevent the active page from being changed.              * @param {Ext.PagingToolbar} this              * @param {Object} params An object hash of the parameters which the PagingToolbar will send when              * loading the required page. This will contain:<ul>              * <li><code>start</code> : Number <div class="sub-desc">The starting row number for the next page of records to              * be retrieved from the server</div></li>              * <li><code>limit</code> : Number <div class="sub-desc">The number of records to be retrieved from the server</div></li>              * </ul>              * <p>(note: the names of the <b>start</b> and <b>limit</b> properties are determined              * by the store's {@link Ext.data.Store#paramNames paramNames} property.)</p>              * <p>Parameters may be added as required in the event handler.</p>              */             'beforechange'         );         this.on('afterlayout', this.onFirstLayout, this, {single: true});         this.cursor = 0;         this.bindStore(this.store);     },     // private     onFirstLayout : function(){         if(this.dsLoaded){             this.onLoad.apply(this, this.dsLoaded);         }     },     // private     updateInfo : function(){         if(this.displayItem){             var count = this.store.getCount();             var msg = count == 0 ?                 this.emptyMsg :                 String.format(                     this.displayMsg,                     this.cursor+1, this.cursor+count, this.store.getTotalCount()                 );             this.displayItem.setText(msg);         }     },     // private     onLoad : function(store, r, o){         if(!this.rendered){             this.dsLoaded = [store, r, o];             return;         }         var p = this.getParams();         this.cursor = (o.params && o.params[p.start]) ? o.params[p.start] : 0;         var d = this.getPageData(), ap = d.activePage, ps = d.pages;         this.afterTextItem.setText(String.format(this.afterPageText, d.pages));         this.inputItem.setValue(ap);         this.first.setDisabled(ap == 1);         this.prev.setDisabled(ap == 1);         this.next.setDisabled(ap == ps);         this.last.setDisabled(ap == ps);         this.refresh.enable();         this.updateInfo();         this.fireEvent('change', this, d);     },     // private     getPageData : function(){         var total = this.store.getTotalCount();         return {             total : total,             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)         };     },     /**      * Change the active page      * @param {Integer} page The page to display      */     changePage : function(page){         this.doLoad(((page-1) * this.pageSize).constrain(0, this.store.getTotalCount()));     },     // private     onLoadError : function(){         if(!this.rendered){             return;         }         this.refresh.enable();     },     // private     readPage : function(d){         var v = this.inputItem.getValue(), pageNum;         if (!v || isNaN(pageNum = parseInt(v, 10))) {             this.inputItem.setValue(d.activePage);             return false;         }         return pageNum;     },     onPagingFocus : function(){         this.inputItem.select();     },     //private     onPagingBlur : function(e){         this.inputItem.setValue(this.getPageData().activePage);     },     // private     onPagingKeyDown : function(field, e){         var k = e.getKey(), d = this.getPageData(), pageNum;         if (k == e.RETURN) {             e.stopEvent();             pageNum = this.readPage(d);             if(pageNum !== false){                 pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;                 this.doLoad(pageNum * this.pageSize);             }         }else if (k == e.HOME || k == e.END){             e.stopEvent();             pageNum = k == e.HOME ? 1 : d.pages;             field.setValue(pageNum);         }else if (k == e.UP || k == e.PAGEUP || k == e.DOWN || k == e.PAGEDOWN){             e.stopEvent();             if((pageNum = this.readPage(d))){                 var increment = e.shiftKey ? 10 : 1;                 if(k == e.DOWN || k == e.PAGEDOWN){                     increment *= -1;                 }                 pageNum += increment;                 if(pageNum >= 1 & pageNum <= d.pages){                     field.setValue(pageNum);                 }             }         }     },     // private     getParams : function(){         //retain backwards compat, allow params on the toolbar itself, if they exist.         return this.paramNames || this.store.paramNames;     },     // private     beforeLoad : function(){         if(this.rendered && this.refresh){             this.refresh.disable();         }     },     // private     doLoad : function(start){         var o = {}, pn = this.getParams();         o[pn.start] = start;         o[pn.limit] = this.pageSize;         if(this.fireEvent('beforechange', this, o) !== false){             this.store.load({params:o});         }     },     /**      * Move to the first page, has the same effect as clicking the 'first' button.      */     moveFirst : function(){         this.doLoad(0);     },     /**      * Move to the previous page, has the same effect as clicking the 'previous' button.      */     movePrevious : function(){         this.doLoad(Math.max(0, this.cursor-this.pageSize));     },     /**      * Move to the next page, has the same effect as clicking the 'next' button.      */     moveNext : function(){         this.doLoad(this.cursor+this.pageSize);     },     /**      * Move to the last page, has the same effect as clicking the 'last' button.      */     moveLast : function(){         var total = this.store.getTotalCount(),             extra = total % this.pageSize;         this.doLoad(extra ? (total - extra) : total - this.pageSize);     },     /**      * Refresh the current page, has the same effect as clicking the 'refresh' button.      */     refresh : function(){         this.doLoad(this.cursor);     },     /**      * Binds the paging toolbar to the specified {@link Ext.data.Store}      * @param {Store} store The store to bind to this toolbar      * @param {Boolean} initial (Optional) true to not remove listeners      */     bindStore : function(store, initial){         var doLoad;         if(!initial && this.store){             this.store.un('beforeload', this.beforeLoad, this);             this.store.un('load', this.onLoad, this);             this.store.un('exception', this.onLoadError, this);             if(store !== this.store && this.store.autoDestroy){                 this.store.destroy();             }         }         if(store){             store = Ext.StoreMgr.lookup(store);             store.on({                 scope: this,                 beforeload: this.beforeLoad,                 load: this.onLoad,                 exception: this.onLoadError             });             doLoad = store.getCount() > 0;         }         this.store = store;         if(doLoad){             this.onLoad(store, null, {});         }     },     /**      * Unbinds the paging toolbar from the specified {@link Ext.data.Store} <b>(deprecated)</b>      * @param {Ext.data.Store} store The data store to unbind      */     unbind : function(store){         this.bindStore(null);     },     /**      * Binds the paging toolbar to the specified {@link Ext.data.Store} <b>(deprecated)</b>      * @param {Ext.data.Store} store The data store to bind      */     bind : function(store){         this.bindStore(store);     },     // private     onDestroy : function(){         this.bindStore(null);         Ext.PagingToolbar.superclass.onDestroy.call(this);     } }); })(); Ext.reg('paging', Ext.PagingToolbar);/**
  1104.  * @class Ext.History
  1105.  * @extends Ext.util.Observable
  1106.  * History management component that allows you to register arbitrary tokens that signify application
  1107.  * history state on navigation actions.  You can then handle the history {@link #change} event in order
  1108.  * to reset your application UI to the appropriate state when the user navigates forward or backward through
  1109.  * the browser history stack.
  1110.  * @singleton
  1111.  */
  1112. Ext.History = (function () {
  1113.     var iframe, hiddenField;
  1114.     var ready = false;
  1115.     var currentToken;
  1116.     function getHash() {
  1117.         var href = top.location.href, i = href.indexOf("#");
  1118.         return i >= 0 ? href.substr(i + 1) : null;
  1119.     }
  1120.     function doSave() {
  1121.         hiddenField.value = currentToken;
  1122.     }
  1123.     function handleStateChange(token) {
  1124.         currentToken = token;
  1125.         Ext.History.fireEvent('change', token);
  1126.     }
  1127.     function updateIFrame (token) {
  1128.         var html = ['<html><body><div id="state">',token,'</div></body></html>'].join('');
  1129.         try {
  1130.             var doc = iframe.contentWindow.document;
  1131.             doc.open();
  1132.             doc.write(html);
  1133.             doc.close();
  1134.             return true;
  1135.         } catch (e) {
  1136.             return false;
  1137.         }
  1138.     }
  1139.     function checkIFrame() {
  1140.         if (!iframe.contentWindow || !iframe.contentWindow.document) {
  1141.             setTimeout(checkIFrame, 10);
  1142.             return;
  1143.         }
  1144.         var doc = iframe.contentWindow.document;
  1145.         var elem = doc.getElementById("state");
  1146.         var token = elem ? elem.innerText : null;
  1147.         var hash = getHash();
  1148.         setInterval(function () {
  1149.             doc = iframe.contentWindow.document;
  1150.             elem = doc.getElementById("state");
  1151.             var newtoken = elem ? elem.innerText : null;
  1152.             var newHash = getHash();
  1153.             if (newtoken !== token) {
  1154.                 token = newtoken;
  1155.                 handleStateChange(token);
  1156.                 top.location.hash = token;
  1157.                 hash = token;
  1158.                 doSave();
  1159.             } else if (newHash !== hash) {
  1160.                 hash = newHash;
  1161.                 updateIFrame(newHash);
  1162.             }
  1163.         }, 50);
  1164.         ready = true;
  1165.         Ext.History.fireEvent('ready', Ext.History);
  1166.     }
  1167.     function startUp() {
  1168.         currentToken = hiddenField.value ? hiddenField.value : getHash();
  1169.         if (Ext.isIE) {
  1170.             checkIFrame();
  1171.         } else {
  1172.             var hash = getHash();
  1173.             setInterval(function () {
  1174.                 var newHash = getHash();
  1175.                 if (newHash !== hash) {
  1176.                     hash = newHash;
  1177.                     handleStateChange(hash);
  1178.                     doSave();
  1179.                 }
  1180.             }, 50);
  1181.             ready = true;
  1182.             Ext.History.fireEvent('ready', Ext.History);
  1183.         }
  1184.     }
  1185.     return {
  1186.         /**
  1187.          * The id of the hidden field required for storing the current history token.
  1188.          * @type String
  1189.          * @property
  1190.          */
  1191.         fieldId: 'x-history-field',
  1192.         /**
  1193.          * The id of the iframe required by IE to manage the history stack.
  1194.          * @type String
  1195.          * @property
  1196.          */
  1197.         iframeId: 'x-history-frame',
  1198.         
  1199.         events:{},
  1200.         /**
  1201.          * Initialize the global History instance.
  1202.          * @param {Boolean} onReady (optional) A callback function that will be called once the history
  1203.          * component is fully initialized.
  1204.          * @param {Object} scope (optional) The callback scope
  1205.          */
  1206.         init: function (onReady, scope) {
  1207.             if(ready) {
  1208.                 Ext.callback(onReady, scope, [this]);
  1209.                 return;
  1210.             }
  1211.             if(!Ext.isReady){
  1212.                 Ext.onReady(function(){
  1213.                     Ext.History.init(onReady, scope);
  1214.                 });
  1215.                 return;
  1216.             }
  1217.             hiddenField = Ext.getDom(Ext.History.fieldId);
  1218.             if (Ext.isIE) {
  1219.                 iframe = Ext.getDom(Ext.History.iframeId);
  1220.             }
  1221.             this.addEvents('ready', 'change');
  1222.             if(onReady){
  1223.                 this.on('ready', onReady, scope, {single:true});
  1224.             }
  1225.             startUp();
  1226.         },
  1227.         /**
  1228.          * Add a new token to the history stack. This can be any arbitrary value, although it would
  1229.          * commonly be the concatenation of a component id and another id marking the specifc history
  1230.          * state of that component.  Example usage:
  1231.          * <pre><code>
  1232. // Handle tab changes on a TabPanel
  1233. tabPanel.on('tabchange', function(tabPanel, tab){
  1234.     Ext.History.add(tabPanel.id + ':' + tab.id);
  1235. });
  1236. </code></pre>
  1237.          * @param {String} token The value that defines a particular application-specific history state
  1238.          * @param {Boolean} preventDuplicates When true, if the passed token matches the current token
  1239.          * it will not save a new history step. Set to false if the same state can be saved more than once
  1240.          * at the same history stack location (defaults to true).
  1241.          */
  1242.         add: function (token, preventDup) {
  1243.             if(preventDup !== false){
  1244.                 if(this.getToken() == token){
  1245.                     return true;
  1246.                 }
  1247.             }
  1248.             if (Ext.isIE) {
  1249.                 return updateIFrame(token);
  1250.             } else {
  1251.                 top.location.hash = token;
  1252.                 return true;
  1253.             }
  1254.         },
  1255.         /**
  1256.          * Programmatically steps back one step in browser history (equivalent to the user pressing the Back button).
  1257.          */
  1258.         back: function(){
  1259.             history.go(-1);
  1260.         },
  1261.         /**
  1262.          * Programmatically steps forward one step in browser history (equivalent to the user pressing the Forward button).
  1263.          */
  1264.         forward: function(){
  1265.             history.go(1);
  1266.         },
  1267.         /**
  1268.          * Retrieves the currently-active history token.
  1269.          * @return {String} The token
  1270.          */
  1271.         getToken: function() {
  1272.             return ready ? currentToken : getHash();
  1273.         }
  1274.     };
  1275. })();
  1276. Ext.apply(Ext.History, new Ext.util.Observable());/**
  1277.  * @class Ext.Tip
  1278.  * @extends Ext.Panel
  1279.  * This is the base class for {@link Ext.QuickTip} and {@link Ext.Tooltip} that provides the basic layout and
  1280.  * positioning that all tip-based classes require. This class can be used directly for simple, statically-positioned
  1281.  * tips that are displayed programmatically, or it can be extended to provide custom tip implementations.
  1282.  * @constructor
  1283.  * Create a new Tip
  1284.  * @param {Object} config The configuration options
  1285.  */
  1286. Ext.Tip = Ext.extend(Ext.Panel, {
  1287.     /**
  1288.      * @cfg {Boolean} closable True to render a close tool button into the tooltip header (defaults to false).
  1289.      */
  1290.     /**
  1291.      * @cfg {Number} width
  1292.      * Width in pixels of the tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
  1293.      * {@link #minWidth} or {@link #maxWidth}.  The maximum supported value is 500.
  1294.      */
  1295.     /**
  1296.      * @cfg {Number} minWidth The minimum width of the tip in pixels (defaults to 40).
  1297.      */
  1298.     minWidth : 40,
  1299.     /**
  1300.      * @cfg {Number} maxWidth The maximum width of the tip in pixels (defaults to 300).  The maximum supported value is 500.
  1301.      */
  1302.     maxWidth : 300,
  1303.     /**
  1304.      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
  1305.      * for bottom-right shadow (defaults to "sides").
  1306.      */
  1307.     shadow : "sides",
  1308.     /**
  1309.      * @cfg {String} defaultAlign <b>Experimental</b>. The default {@link Ext.Element#alignTo} anchor position value
  1310.      * for this tip relative to its element of origin (defaults to "tl-bl?").
  1311.      */
  1312.     defaultAlign : "tl-bl?",
  1313.     autoRender: true,
  1314.     quickShowInterval : 250,
  1315.     // private panel overrides
  1316.     frame:true,
  1317.     hidden:true,
  1318.     baseCls: 'x-tip',
  1319.     floating:{shadow:true,shim:true,useDisplay:true,constrain:false},
  1320.     autoHeight:true,
  1321.     closeAction: 'hide',
  1322.     // private
  1323.     initComponent : function(){
  1324.         Ext.Tip.superclass.initComponent.call(this);
  1325.         if(this.closable && !this.title){
  1326.             this.elements += ',header';
  1327.         }
  1328.     },
  1329.     // private
  1330.     afterRender : function(){
  1331.         Ext.Tip.superclass.afterRender.call(this);
  1332.         if(this.closable){
  1333.             this.addTool({
  1334.                 id: 'close',
  1335.                 handler: this[this.closeAction],
  1336.                 scope: this
  1337.             });
  1338.         }
  1339.     },
  1340.     /**
  1341.      * Shows this tip at the specified XY position.  Example usage:
  1342.      * <pre><code>
  1343. // Show the tip at x:50 and y:100
  1344. tip.showAt([50,100]);
  1345. </code></pre>
  1346.      * @param {Array} xy An array containing the x and y coordinates
  1347.      */
  1348.     showAt : function(xy){
  1349.         Ext.Tip.superclass.show.call(this);
  1350.         if(this.measureWidth !== false && (!this.initialConfig || typeof this.initialConfig.width != 'number')){
  1351.             this.doAutoWidth();
  1352.         }
  1353.         if(this.constrainPosition){
  1354.             xy = this.el.adjustForConstraints(xy);
  1355.         }
  1356.         this.setPagePosition(xy[0], xy[1]);
  1357.     },
  1358.     // protected
  1359.     doAutoWidth : function(){
  1360.         var bw = this.body.getTextWidth();
  1361.         if(this.title){
  1362.             bw = Math.max(bw, this.header.child('span').getTextWidth(this.title));
  1363.         }
  1364.         bw += this.getFrameWidth() + (this.closable ? 20 : 0) + this.body.getPadding("lr");
  1365.         this.setWidth(bw.constrain(this.minWidth, this.maxWidth));
  1366.         
  1367.         // IE7 repaint bug on initial show
  1368.         if(Ext.isIE7 && !this.repainted){
  1369.             this.el.repaint();
  1370.             this.repainted = true;
  1371.         }
  1372.     },
  1373.     /**
  1374.      * <b>Experimental</b>. Shows this tip at a position relative to another element using a standard {@link Ext.Element#alignTo}
  1375.      * anchor position value.  Example usage:
  1376.      * <pre><code>
  1377. // Show the tip at the default position ('tl-br?')
  1378. tip.showBy('my-el');
  1379. // Show the tip's top-left corner anchored to the element's top-right corner
  1380. tip.showBy('my-el', 'tl-tr');
  1381. </code></pre>
  1382.      * @param {Mixed} el An HTMLElement, Ext.Element or string id of the target element to align to
  1383.      * @param {String} position (optional) A valid {@link Ext.Element#alignTo} anchor position (defaults to 'tl-br?' or
  1384.      * {@link #defaultAlign} if specified).
  1385.      */
  1386.     showBy : function(el, pos){
  1387.         if(!this.rendered){
  1388.             this.render(Ext.getBody());
  1389.         }
  1390.         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign));
  1391.     },
  1392.     initDraggable : function(){
  1393.         this.dd = new Ext.Tip.DD(this, typeof this.draggable == 'boolean' ? null : this.draggable);
  1394.         this.header.addClass('x-tip-draggable');
  1395.     }
  1396. });
  1397. // private - custom Tip DD implementation
  1398. Ext.Tip.DD = function(tip, config){
  1399.     Ext.apply(this, config);
  1400.     this.tip = tip;
  1401.     Ext.Tip.DD.superclass.constructor.call(this, tip.el.id, 'WindowDD-'+tip.id);
  1402.     this.setHandleElId(tip.header.id);
  1403.     this.scroll = false;
  1404. };
  1405. Ext.extend(Ext.Tip.DD, Ext.dd.DD, {
  1406.     moveOnly:true,
  1407.     scroll:false,
  1408.     headerOffsets:[100, 25],
  1409.     startDrag : function(){
  1410.         this.tip.el.disableShadow();
  1411.     },
  1412.     endDrag : function(e){
  1413.         this.tip.el.enableShadow(true);
  1414.     }
  1415. });/**
  1416.  * @class Ext.ToolTip
  1417.  * @extends Ext.Tip
  1418.  * A standard tooltip implementation for providing additional information when hovering over a target element.
  1419.  * @constructor
  1420.  * Create a new Tooltip
  1421.  * @param {Object} config The configuration options
  1422.  */
  1423. Ext.ToolTip = Ext.extend(Ext.Tip, {
  1424.     /**
  1425.      * When a Tooltip is configured with the {@link #delegate} option to cause selected child elements of the {@link #target}
  1426.      * Element to each trigger a seperate show event, this property is set to the DOM element which triggered the show.
  1427.      * @type DOMElement
  1428.      * @property triggerElement
  1429.      */
  1430.     /**
  1431.      * @cfg {Mixed} target The target HTMLElement, Ext.Element or id to monitor for mouseover events to trigger
  1432.      * showing this ToolTip.
  1433.      */
  1434.     /**
  1435.      * @cfg {Boolean} autoHide True to automatically hide the tooltip after the mouse exits the target element
  1436.      * or after the {@link #dismissDelay} has expired if set (defaults to true).  If {@link closable} = true a close
  1437.      * tool button will be rendered into the tooltip header.
  1438.      */
  1439.     /**
  1440.      * @cfg {Number} showDelay Delay in milliseconds before the tooltip displays after the mouse enters the
  1441.      * target element (defaults to 500)
  1442.      */
  1443.     showDelay: 500,
  1444.     /**
  1445.      * @cfg {Number} hideDelay Delay in milliseconds after the mouse exits the target element but before the
  1446.      * tooltip actually hides (defaults to 200).  Set to 0 for the tooltip to hide immediately.
  1447.      */
  1448.     hideDelay: 200,
  1449.     /**
  1450.      * @cfg {Number} dismissDelay Delay in milliseconds before the tooltip automatically hides (defaults to 5000).
  1451.      * To disable automatic hiding, set dismissDelay = 0.
  1452.      */
  1453.     dismissDelay: 5000,
  1454.     /**
  1455.      * @cfg {Array} mouseOffset An XY offset from the mouse position where the tooltip should be shown (defaults to [15,18]).
  1456.      */
  1457.     /**
  1458.      * @cfg {Boolean} trackMouse True to have the tooltip follow the mouse as it moves over the target element (defaults to false).
  1459.      */
  1460.     trackMouse : false,
  1461.     /**
  1462.      * @cfg {Boolean} anchorToTarget True to anchor the tooltip to the target element, false to
  1463.      * anchor it relative to the mouse coordinates (defaults to true).  When anchorToTarget is
  1464.      * true, use {@link #defaultAlign} to control tooltip alignment to the target element.  When
  1465.      * anchorToTarget is false, use {@link #anchorPosition} instead to control alignment.
  1466.      */
  1467.     anchorToTarget: true,
  1468.     /**
  1469.      * @cfg {Number} anchorOffset A numeric pixel value used to offset the default position of the
  1470.      * anchor arrow (defaults to 0).  When the anchor position is on the top or bottom of the tooltip,
  1471.      * anchorOffset will be used as a horizontal offset.  Likewise, when the anchor position is on the
  1472.      * left or right side, anchorOffset will be used as a vertical offset.
  1473.      */
  1474.     anchorOffset: 0,
  1475.     /**
  1476.      * @cfg {String} delegate <p>Optional. A {@link Ext.DomQuery DomQuery} selector which allows selection of individual elements
  1477.      * within the {@link #target} element to trigger showing and hiding the ToolTip as the mouse moves within the target.</p>
  1478.      * <p>When specified, the child element of the target which caused a show event is placed into the {@link #triggerElement} property
  1479.      * before the ToolTip is shown.</p>
  1480.      * <p>This may be useful when a Component has regular, repeating elements in it, each of which need a Tooltip which contains
  1481.      * information specific to that element. For example:</p><pre><code>
  1482. var myGrid = new Ext.grid.gridPanel(gridConfig);
  1483. myGrid.on('render', function(grid) {
  1484.     var store = grid.getStore();  // Capture the Store.
  1485.     var view = grid.getView();    // Capture the GridView.
  1486.     myGrid.tip = new Ext.ToolTip({
  1487.         target: view.mainBody,    // The overall target element.
  1488.         delegate: '.x-grid3-row', // Each grid row causes its own seperate show and hide.
  1489.         trackMouse: true,         // Moving within the row should not hide the tip.
  1490.         renderTo: document.body,  // Render immediately so that tip.body can be referenced prior to the first show.
  1491.         listeners: {              // Change content dynamically depending on which element triggered the show.
  1492.             beforeshow: function updateTipBody(tip) {
  1493.                 var rowIndex = view.findRowIndex(tip.triggerElement);
  1494.                 tip.body.dom.innerHTML = "Over Record ID " + store.getAt(rowIndex).id;
  1495.             }
  1496.         }
  1497.     });
  1498. });</code></pre>
  1499.      */
  1500.     // private
  1501.     targetCounter: 0,
  1502.     constrainPosition: false,
  1503.     // private
  1504.     initComponent: function(){
  1505.         Ext.ToolTip.superclass.initComponent.call(this);
  1506.         this.lastActive = new Date();
  1507.         this.initTarget(this.target);
  1508.         this.origAnchor = this.anchor;
  1509.     },
  1510.     // private
  1511.     onRender : function(ct, position){
  1512.         Ext.ToolTip.superclass.onRender.call(this, ct, position);
  1513.         this.anchorCls = 'x-tip-anchor-' + this.getAnchorPosition();
  1514.         this.anchorEl = this.el.createChild({
  1515.             cls: 'x-tip-anchor ' + this.anchorCls
  1516.         });
  1517.     },
  1518.     // private