Upgrade to ExtJS 4.0.7 - Released 10/19/2011
[extjs.git] / docs / source / Button.html
index 7e34b47..9b2f052 100644 (file)
-<html>\r
-<head>\r
-  <title>The source code</title>\r
-    <link href="../resources/prettify/prettify.css" type="text/css" rel="stylesheet" />\r
-    <script type="text/javascript" src="../resources/prettify/prettify.js"></script>\r
-</head>\r
-<body  onload="prettyPrint();">\r
-    <pre class="prettyprint lang-js"><div id="cls-Ext.Button"></div>/**
- * @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 {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
+<!DOCTYPE html>
+<html>
+<head>
+  <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
+  <title>The source code</title>
+  <link href="../resources/prettify/prettify.css" type="text/css" rel="stylesheet" />
+  <script type="text/javascript" src="../resources/prettify/prettify.js"></script>
+  <style type="text/css">
+    .highlight { display: block; background-color: #ddd; }
+  </style>
+  <script type="text/javascript">
+    function highlight() {
+      document.getElementById(location.hash.replace(/#/, "")).className = "highlight";
+    }
+  </script>
+</head>
+<body onload="prettyPrint(); highlight();">
+  <pre class="prettyprint lang-js"><span id='Ext-button-Button'>/**
+</span> * @docauthor Robert Dougan &lt;rob@sencha.com&gt;
+ *
+ * Create simple buttons with this component. Customisations include {@link #iconAlign aligned}
+ * {@link #iconCls icons}, {@link #menu dropdown menus}, {@link #tooltip tooltips}
+ * and {@link #scale sizing options}. Specify a {@link #handler handler} to run code when
+ * a user clicks the button, or use {@link #listeners listeners} for other events such as
+ * {@link #mouseover mouseover}. Example usage:
+ *
+ *     @example
+ *     Ext.create('Ext.Button', {
+ *         text: 'Click me',
+ *         renderTo: Ext.getBody(),
+ *         handler: function() {
+ *             alert('You clicked the button!')
+ *         }
+ *     });
+ *
+ * The {@link #handler} configuration can also be updated dynamically using the {@link #setHandler}
+ * method.  Example usage:
+ *
+ *     @example
+ *     Ext.create('Ext.Button', {
+ *         text    : 'Dynamic Handler Button',
+ *         renderTo: Ext.getBody(),
+ *         handler : function() {
+ *             // this button will spit out a different number every time you click it.
+ *             // so firstly we must check if that number is already set:
+ *             if (this.clickCount) {
+ *                 // looks like the property is already set, so lets just add 1 to that number and alert the user
+ *                 this.clickCount++;
+ *                 alert('You have clicked the button &quot;' + this.clickCount + '&quot; times.\n\nTry clicking it again..');
+ *             } else {
+ *                 // if the clickCount property is not set, we will set it and alert the user
+ *                 this.clickCount = 1;
+ *                 alert('You just clicked the button for the first time!\n\nTry pressing it again..');
+ *             }
+ *         }
+ *     });
+ *
+ * A button within a container:
+ *
+ *     @example
+ *     Ext.create('Ext.Container', {
+ *         renderTo: Ext.getBody(),
+ *         items   : [
+ *             {
+ *                 xtype: 'button',
+ *                 text : 'My Button'
+ *             }
+ *         ]
+ *     });
+ *
+ * A useful option of Button is the {@link #scale} configuration. This configuration has three different options:
+ *
+ * - `'small'`
+ * - `'medium'`
+ * - `'large'`
+ *
+ * Example usage:
+ *
+ *     @example
+ *     Ext.create('Ext.Button', {
+ *         renderTo: document.body,
+ *         text    : 'Click me',
+ *         scale   : 'large'
+ *     });
+ *
+ * Buttons can also be toggled. To enable this, you simple set the {@link #enableToggle} property to `true`.
+ * Example usage:
+ *
+ *     @example
+ *     Ext.create('Ext.Button', {
+ *         renderTo: Ext.getBody(),
+ *         text: 'Click Me',
+ *         enableToggle: true
+ *     });
+ *
+ * You can assign a menu to a button by using the {@link #menu} configuration. This standard configuration
+ * can either be a reference to a {@link Ext.menu.Menu menu} object, a {@link Ext.menu.Menu menu} id or a
+ * {@link Ext.menu.Menu menu} config blob. When assigning a menu to a button, an arrow is automatically
+ * added to the button.  You can change the alignment of the arrow using the {@link #arrowAlign} configuration
+ * on button.  Example usage:
+ *
+ *     @example
+ *     Ext.create('Ext.Button', {
+ *         text      : 'Menu button',
+ *         renderTo  : Ext.getBody(),
+ *         arrowAlign: 'bottom',
+ *         menu      : [
+ *             {text: 'Item 1'},
+ *             {text: 'Item 2'},
+ *             {text: 'Item 3'},
+ *             {text: 'Item 4'}
+ *         ]
+ *     });
+ *
+ * Using listeners, you can easily listen to events fired by any component, using the {@link #listeners}
+ * configuration or using the {@link #addListener} method.  Button has a variety of different listeners:
+ *
+ * - `click`
+ * - `toggle`
+ * - `mouseover`
+ * - `mouseout`
+ * - `mouseshow`
+ * - `menuhide`
+ * - `menutriggerover`
+ * - `menutriggerout`
+ *
+ * Example usage:
+ *
+ *     @example
+ *     Ext.create('Ext.Button', {
+ *         text     : 'Button',
+ *         renderTo : Ext.getBody(),
+ *         listeners: {
+ *             click: function() {
+ *                 // this == the button, as we are in the local scope
+ *                 this.setText('I was clicked!');
+ *             },
+ *             mouseover: function() {
+ *                 // set a new config which says we moused over, if not already set
+ *                 if (!this.mousedOver) {
+ *                     this.mousedOver = true;
+ *                     alert('You moused over a button!\n\nI wont do this again.');
+ *                 }
+ *             }
+ *         }
+ *     });
  */
-Ext.Button = Ext.extend(Ext.BoxComponent, {
-    /**
-     * Read-only. True if this button is hidden
-     * @type Boolean
+Ext.define('Ext.button.Button', {
+
+    /* Begin Definitions */
+    alias: 'widget.button',
+    extend: 'Ext.Component',
+
+    requires: [
+        'Ext.menu.Manager',
+        'Ext.util.ClickRepeater',
+        'Ext.layout.component.Button',
+        'Ext.util.TextMetrics',
+        'Ext.util.KeyMap'
+    ],
+
+    alternateClassName: 'Ext.Button',
+    /* End Definitions */
+
+    isButton: true,
+    componentLayout: 'button',
+
+<span id='Ext-button-Button-property-hidden'>    /**
+</span>     * @property {Boolean} hidden
+     * True if this button is hidden. Read-only.
      */
-    hidden : false,
-    /**
-     * Read-only. True if this button is disabled
-     * @type Boolean
+    hidden: false,
+
+<span id='Ext-button-Button-property-disabled'>    /**
+</span>     * @property {Boolean} disabled
+     * True if this button is disabled. Read-only.
      */
-    disabled : false,
-    /**
-     * Read-only. True if this button is pressed (only if enableToggle = true)
-     * @type Boolean
+    disabled: false,
+
+<span id='Ext-button-Button-property-pressed'>    /**
+</span>     * @property {Boolean} pressed
+     * True if this button is pressed (only if enableToggle = true). Read-only.
      */
-    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
+    pressed: false,
+
+<span id='Ext-button-Button-cfg-text'>    /**
+</span>     * @cfg {String} text
+     * The button text to be used as innerHTML (html tags are accepted).
      */
 
-    /**
-     * @cfg {Number} tabIndex Set a DOM tabIndex for this button (defaults to undefined)
+<span id='Ext-button-Button-cfg-icon'>    /**
+</span>     * @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 {Boolean} allowDepress
-     * False to not allow a pressed Button to be depressed (defaults to undefined). Only valid when {@link #enableToggle} is true.
+<span id='Ext-button-Button-cfg-handler'>    /**
+</span>     * @cfg {Function} handler
+     * A function called when the button is clicked (can be used instead of click event).
+     * @cfg {Ext.button.Button} handler.button This button.
+     * @cfg {Ext.EventObject} handler.e The click event.
      */
 
-    /**
-     * @cfg {Boolean} enableToggle
-     * True to enable pressed/not pressed toggling (defaults to false)
+<span id='Ext-button-Button-cfg-minWidth'>    /**
+</span>     * @cfg {Number} minWidth
+     * The minimum width for this button (used to give a set of buttons a common width).
+     * See also {@link Ext.panel.Panel}.{@link Ext.panel.Panel#minButtonWidth minButtonWidth}.
      */
-    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>
+
+<span id='Ext-button-Button-cfg-tooltip'>    /**
+</span>     * @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.
+     */
+
+<span id='Ext-button-Button-cfg-hidden'>    /**
+</span>     * @cfg {Boolean} [hidden=false]
+     * True to start hidden.
      */
-    /**
-     * @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).
+
+<span id='Ext-button-Button-cfg-disabled'>    /**
+</span>     * @cfg {Boolean} [disabled=true]
+     * True to start disabled.
      */
-    /**
-     * @cfg {String} menuAlign
-     * The position to align the menu to (see {@link Ext.Element#alignTo} for more details, defaults to 'tl-bl?').
+
+<span id='Ext-button-Button-cfg-pressed'>    /**
+</span>     * @cfg {Boolean} [pressed=false]
+     * True to start pressed (only if enableToggle = true)
      */
-    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>.
+<span id='Ext-button-Button-cfg-toggleGroup'>    /**
+</span>     * @cfg {String} toggleGroup
+     * The group this toggle button is a member of (only 1 per group can be pressed)
      */
-    /**
-     * @cfg {String} iconCls
-     * A css class which sets a background image to be used as the icon for this button
+
+<span id='Ext-button-Button-cfg-repeat'>    /**
+</span>     * @cfg {Boolean/Object} [repeat=false]
+     * True to repeat fire the click event while the mouse is down. This can also be a
+     * {@link Ext.util.ClickRepeater ClickRepeater} config object.
      */
-    /**
-     * @cfg {String} type
-     * submit, reset or button - defaults to 'button'
+
+<span id='Ext-button-Button-cfg-tabIndex'>    /**
+</span>     * @cfg {Number} tabIndex
+     * Set a DOM tabIndex for this 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 {Object} scope The scope (<tt><b>this</b></tt> reference) in which the
-     * <code>{@link #handler}</code> and <code>{@link #toggleHandler}</code> is
-     * executed. Defaults to this Button.
-     */
-
-    /**
-     * @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
+<span id='Ext-button-Button-cfg-allowDepress'>    /**
+</span>     * @cfg {Boolean} [allowDepress=true]
+     * False to not allow a pressed Button to be depressed. Only valid when {@link #enableToggle} is true.
+     */
+
+<span id='Ext-button-Button-cfg-enableToggle'>    /**
+</span>     * @cfg {Boolean} [enableToggle=false]
+     * True to enable pressed/not pressed toggling.
+     */
+    enableToggle: false,
+
+<span id='Ext-button-Button-cfg-toggleHandler'>    /**
+</span>     * @cfg {Function} toggleHandler
+     * Function called when a Button with {@link #enableToggle} set to true is clicked.
+     * @cfg {Ext.button.Button} toggleHandler.button This button.
+     * @cfg {Boolean} toggleHandler.state The next state of the Button, true means pressed.
+     */
+
+<span id='Ext-button-Button-cfg-menu'>    /**
+</span>     * @cfg {Ext.menu.Menu/String/Object} menu
+     * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob.
+     */
+
+<span id='Ext-button-Button-cfg-menuAlign'>    /**
+</span>     * @cfg {String} menuAlign
+     * The position to align the menu to (see {@link Ext.Element#alignTo} for more details).
+     */
+    menuAlign: 'tl-bl?',
+
+<span id='Ext-button-Button-cfg-textAlign'>    /**
+</span>     * @cfg {String} textAlign
+     * The text alignment for this button (center, left, right).
+     */
+    textAlign: 'center',
+
+<span id='Ext-button-Button-cfg-overflowText'>    /**
+</span>     * @cfg {String} overflowText
+     * If used in a {@link Ext.toolbar.Toolbar Toolbar}, the text to be used if this item is shown in the overflow menu.
+     * See also {@link Ext.toolbar.Item}.`{@link Ext.toolbar.Item#overflowText overflowText}`.
+     */
+
+<span id='Ext-button-Button-cfg-iconCls'>    /**
+</span>     * @cfg {String} iconCls
+     * A css class which sets a background image to be used as the icon for this button.
+     */
+
+<span id='Ext-button-Button-cfg-type'>    /**
+</span>     * @cfg {String} type
+     * The type of `&lt;input&gt;` to create: submit, reset or button.
+     */
+    type: 'button',
+
+<span id='Ext-button-Button-cfg-clickEvent'>    /**
+</span>     * @cfg {String} clickEvent
+     * The DOM event that will fire the handler of the button. This can be any valid event name (dblclick, contextmenu).
+     */
+    clickEvent: 'click',
+
+<span id='Ext-button-Button-cfg-preventDefault'>    /**
+</span>     * @cfg {Boolean} preventDefault
+     * True to prevent the default action when the {@link #clickEvent} is processed.
+     */
+    preventDefault: true,
+
+<span id='Ext-button-Button-cfg-handleMouseEvents'>    /**
+</span>     * @cfg {Boolean} handleMouseEvents
+     * False to disable visual cues on mouseover, mouseout and mousedown.
+     */
+    handleMouseEvents: true,
+
+<span id='Ext-button-Button-cfg-tooltipType'>    /**
+</span>     * @cfg {String} tooltipType
+     * The type of tooltip to use. Either 'qtip' for QuickTips or 'title' for title attribute.
+     */
+    tooltipType: 'qtip',
+
+<span id='Ext-button-Button-cfg-baseCls'>    /**
+</span>     * @cfg {String} [baseCls='x-btn']
+     * The base CSS class to add to all buttons.
+     */
+    baseCls: Ext.baseCSSPrefix + 'btn',
+
+<span id='Ext-button-Button-cfg-pressedCls'>    /**
+</span>     * @cfg {String} pressedCls
+     * The CSS class to add to a button when it is in the pressed state.
+     */
+    pressedCls: 'pressed',
+
+<span id='Ext-button-Button-cfg-overCls'>    /**
+</span>     * @cfg {String} overCls
+     * The CSS class to add to a button when it is in the over (hovered) state.
+     */
+    overCls: 'over',
+
+<span id='Ext-button-Button-cfg-focusCls'>    /**
+</span>     * @cfg {String} focusCls
+     * The CSS class to add to a button when it is in the focussed state.
+     */
+    focusCls: 'focus',
+
+<span id='Ext-button-Button-cfg-menuActiveCls'>    /**
+</span>     * @cfg {String} menuActiveCls
+     * The CSS class to add to a button when it's menu is active.
+     */
+    menuActiveCls: 'menu-active',
+
+<span id='Ext-button-Button-cfg-href'>    /**
+</span>     * @cfg {String} href
+     * The URL to visit when the button is clicked. Specifying this config is equivalent to specifying:
+     *
+     *     handler: function() { window.location = &quot;http://www.sencha.com&quot; }
+     */
+
+<span id='Ext-button-Button-cfg-baseParams'>    /**
+</span>     * @cfg {Object} baseParams
+     * An object literal of parameters to pass to the url when the {@link #href} property is specified.
+     */
+
+<span id='Ext-button-Button-cfg-params'>    /**
+</span>     * @cfg {Object} params
+     * An object literal of parameters to pass to the url when the {@link #href} property is specified. Any params
+     * override {@link #baseParams}. New params can be set using the {@link #setParams} method.
+     */
+
+    ariaRole: 'button',
+
+    // inherited
+    renderTpl:
+        '&lt;em id=&quot;{id}-btnWrap&quot; class=&quot;{splitCls}&quot;&gt;' +
+            '&lt;tpl if=&quot;href&quot;&gt;' +
+                '&lt;a id=&quot;{id}-btnEl&quot; href=&quot;{href}&quot; target=&quot;{target}&quot;&lt;tpl if=&quot;tabIndex&quot;&gt; tabIndex=&quot;{tabIndex}&quot;&lt;/tpl&gt; role=&quot;link&quot;&gt;' +
+                    '&lt;span id=&quot;{id}-btnInnerEl&quot; class=&quot;{baseCls}-inner&quot;&gt;' +
+                        '{text}' +
+                    '&lt;/span&gt;' +
+                        '&lt;span id=&quot;{id}-btnIconEl&quot; class=&quot;{baseCls}-icon&quot;&gt;&lt;/span&gt;' +
+                '&lt;/a&gt;' +
+            '&lt;/tpl&gt;' +
+            '&lt;tpl if=&quot;!href&quot;&gt;' +
+                '&lt;button id=&quot;{id}-btnEl&quot; type=&quot;{type}&quot; hidefocus=&quot;true&quot;' +
+                    // the autocomplete=&quot;off&quot; is required to prevent Firefox from remembering
+                    // the button's disabled state between page reloads.
+                    '&lt;tpl if=&quot;tabIndex&quot;&gt; tabIndex=&quot;{tabIndex}&quot;&lt;/tpl&gt; role=&quot;button&quot; autocomplete=&quot;off&quot;&gt;' +
+                    '&lt;span id=&quot;{id}-btnInnerEl&quot; class=&quot;{baseCls}-inner&quot; style=&quot;{innerSpanStyle}&quot;&gt;' +
+                        '{text}' +
+                    '&lt;/span&gt;' +
+                    '&lt;span id=&quot;{id}-btnIconEl&quot; class=&quot;{baseCls}-icon {iconCls}&quot;&gt;&amp;#160;&lt;/span&gt;' +
+                '&lt;/button&gt;' +
+            '&lt;/tpl&gt;' +
+        '&lt;/em&gt;' ,
+
+<span id='Ext-button-Button-cfg-scale'>    /**
+</span>     * @cfg {String} scale
+     * The size of the Button. Three values are allowed:
+     *
+     * - 'small' - Results in the button element being 16px high.
+     * - 'medium' - Results in the button element being 24px high.
+     * - 'large' - Results in the button element being 32px high.
+     */
+    scale: 'small',
+
+<span id='Ext-button-Button-property-allowedScales'>    /**
+</span>     * @private
+     * An array of allowed scales.
+     */
+    allowedScales: ['small', 'medium', 'large'],
+
+<span id='Ext-button-Button-cfg-scope'>    /**
+</span>     * @cfg {Object} scope
+     * The scope (**this** reference) in which the `{@link #handler}` and `{@link #toggleHandler}` is executed.
+     * Defaults to this Button.
+     */
+
+<span id='Ext-button-Button-cfg-iconAlign'>    /**
+</span>     * @cfg {String} iconAlign
+     * The side of the Button box to render the icon. Four values are allowed:
+     *
+     * - 'top'
+     * - 'right'
+     * - 'bottom'
+     * - 'left'
+     */
+    iconAlign: 'left',
+
+<span id='Ext-button-Button-cfg-arrowAlign'>    /**
+</span>     * @cfg {String} arrowAlign
+     * The side of the Button box to render the arrow if the button has an associated {@link #menu}. Two
+     * values are allowed:
+     *
+     * - 'right'
+     * - 'bottom'
+     */
+    arrowAlign: 'right',
+
+<span id='Ext-button-Button-cfg-arrowCls'>    /**
+</span>     * @cfg {String} arrowCls
+     * The className used for the inner arrow element if the button has a menu.
+     */
+    arrowCls: 'arrow',
+
+<span id='Ext-button-Button-property-template'>    /**
+</span>     * @property {Ext.Template} template
+     * A {@link Ext.Template Template} used to create the Button's DOM structure.
+     *
+     * Instances, or subclasses which need a different DOM structure may provide a different template layout in
+     * conjunction with an implementation of {@link #getTemplateArgs}.
+     */
+
+<span id='Ext-button-Button-cfg-cls'>    /**
+</span>     * @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.
+
+<span id='Ext-button-Button-property-menu'>    /**
+</span>     * @property {Ext.menu.Menu} menu
+     * The {@link Ext.menu.Menu Menu} object associated with this Button when configured with the {@link #menu} config
+     * option.
+     */
+
+<span id='Ext-button-Button-cfg-autoWidth'>    /**
+</span>     * @cfg {Boolean} autoWidth
+     * By default, if a width is not specified the button will attempt to stretch horizontally to fit its content. If
+     * the button is being managed by a width sizing layout (hbox, fit, anchor), set this to false to prevent the button
+     * from doing this automatic sizing.
      */
 
-    initComponent : function(){
-        Ext.Button.superclass.initComponent.call(this);
+    maskOnDisable: false,
 
-        this.addEvents(
-            /**
-             * @event click
+    // inherit docs
+    initComponent: function() {
+        var me = this;
+        me.callParent(arguments);
+
+        me.addEvents(
+<span id='Ext-button-Button-event-click'>            /**
+</span>             * @event click
              * Fires when this button is clicked
-             * @param {Button} this
-             * @param {EventObject} e The click event
+             * @param {Ext.button.Button} this
+             * @param {Event} e The click event
              */
             'click',
-            /**
-             * @event toggle
+
+<span id='Ext-button-Button-event-toggle'>            /**
+</span>             * @event toggle
              * Fires when the 'pressed' state of this button changes (only if enableToggle = true)
-             * @param {Button} this
+             * @param {Ext.button.Button} this
              * @param {Boolean} pressed
              */
             'toggle',
-            /**
-             * @event mouseover
+
+<span id='Ext-button-Button-event-mouseover'>            /**
+</span>             * @event mouseover
              * Fires when the mouse hovers over the button
-             * @param {Button} this
+             * @param {Ext.button.Button} this
              * @param {Event} e The event object
              */
             'mouseover',
-            /**
-             * @event mouseout
+
+<span id='Ext-button-Button-event-mouseout'>            /**
+</span>             * @event mouseout
              * Fires when the mouse exits the button
-             * @param {Button} this
+             * @param {Ext.button.Button} this
              * @param {Event} e The event object
              */
             'mouseout',
-            /**
-             * @event menushow
+
+<span id='Ext-button-Button-event-menushow'>            /**
+</span>             * @event menushow
              * If this button has a menu, this event fires when it is shown
-             * @param {Button} this
-             * @param {Menu} menu
+             * @param {Ext.button.Button} this
+             * @param {Ext.menu.Menu} menu
              */
             'menushow',
-            /**
-             * @event menuhide
+
+<span id='Ext-button-Button-event-menuhide'>            /**
+</span>             * @event menuhide
              * If this button has a menu, this event fires when it is hidden
-             * @param {Button} this
-             * @param {Menu} menu
+             * @param {Ext.button.Button} this
+             * @param {Ext.menu.Menu} menu
              */
             'menuhide',
-            /**
-             * @event menutriggerover
+
+<span id='Ext-button-Button-event-menutriggerover'>            /**
+</span>             * @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
+             * @param {Ext.button.Button} this
+             * @param {Ext.menu.Menu} menu
+             * @param {Event} e
              */
             'menutriggerover',
-            /**
-             * @event menutriggerout
+
+<span id='Ext-button-Button-event-menutriggerout'>            /**
+</span>             * @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
+             * @param {Ext.button.Button} this
+             * @param {Ext.menu.Menu} menu
+             * @param {Event} 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';
+
+        if (me.menu) {
+            // Flag that we'll have a splitCls
+            me.split = true;
+
+            // retrieve menu by id or instantiate instance if needed
+            me.menu = Ext.menu.Manager.get(me.menu);
+            me.menu.ownerCt = me;
+        }
+
+        // Accept url as a synonym for href
+        if (me.url) {
+            me.href = me.url;
+        }
+
+        // preventDefault defaults to false for links
+        if (me.href &amp;&amp; !me.hasOwnProperty('preventDefault')) {
+            me.preventDefault = false;
+        }
+
+        if (Ext.isString(me.toggleGroup)) {
+            me.enableToggle = true;
+        }
+
+    },
+
+    // private
+    initAria: function() {
+        this.callParent();
+        var actionEl = this.getActionEl();
+        if (this.menu) {
+            actionEl.dom.setAttribute('aria-haspopup', true);
         }
-        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') : '';
+    // inherit docs
+    getActionEl: function() {
+        return this.btnEl;
+    },
+
+    // inherit docs
+    getFocusEl: function() {
+        return this.btnEl;
     },
 
     // 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();
+    setButtonCls: function() {
+        var me = this,
+            cls = [],
+            btnIconEl = me.btnIconEl,
+            hide = 'x-hide-display';
+
+        if (me.useSetClass) {
+            if (!Ext.isEmpty(me.oldCls)) {
+                me.removeClsWithUI(me.oldCls);
+                me.removeClsWithUI(me.pressedCls);
             }
-            this.template = Ext.Button.buttonTemplate;
+
+            // Check whether the button has an icon or not, and if it has an icon, what is th alignment
+            if (me.iconCls || me.icon) {
+                if (me.text) {
+                    cls.push('icon-text-' + me.iconAlign);
+                } else {
+                    cls.push('icon');
+                }
+                if (btnIconEl) {
+                    btnIconEl.removeCls(hide);
+                }
+            } else {
+                if (me.text) {
+                    cls.push('noicon');
+                }
+                if (btnIconEl) {
+                    btnIconEl.addCls(hide);
+                }
+            }
+
+            me.oldCls = cls;
+            me.addClsWithUI(cls);
+            me.addClsWithUI(me.pressed ? me.pressedCls : null);
         }
+    },
 
-        var btn, targs = this.getTemplateArgs();
+    // private
+    onRender: function(ct, position) {
+        // classNames for the button
+        var me = this,
+            repeater, btn;
+
+        // Apply the renderData to the template args
+        Ext.applyIf(me.renderData, me.getTemplateArgs());
 
-        if(position){
-            btn = this.template.insertBefore(position, targs, true);
-        }else{
-            btn = this.template.append(ct, targs, true);
+        me.addChildEls('btnEl', 'btnWrap', 'btnInnerEl', 'btnIconEl');
+
+        if (me.scale) {
+            me.ui = me.ui + '-' + me.scale;
         }
-        /**
-         * 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);
+        // Render internal structure
+        me.callParent(arguments);
 
-        Ext.ButtonToggleMgr.register(this);
-    },
+        // If it is a split button + has a toolip for the arrow
+        if (me.split &amp;&amp; me.arrowTooltip) {
+            me.arrowEl.dom.setAttribute(me.getTipAttr(), me.arrowTooltip);
+        }
 
-    // private
-    initButtonEl : function(btn, btnEl){
-        this.el = btn;
+        // Add listeners to the focus and blur events on the element
+        me.mon(me.btnEl, {
+            scope: me,
+            focus: me.onFocus,
+            blur : me.onBlur
+        });
+
+        // Set btn as a local variable for easy access
+        btn = me.el;
 
-        if(this.id){
-            this.el.dom.id = this.el.id = this.id;
+        if (me.icon) {
+            me.setIcon(me.icon);
         }
-        if(this.icon){
-            btnEl.setStyle('background-image', 'url(' +this.icon +')');
+
+        if (me.iconCls) {
+            me.setIconCls(me.iconCls);
         }
-        if(this.tabIndex !== undefined){
-            btnEl.dom.tabIndex = this.tabIndex;
+
+        if (me.tooltip) {
+            me.setTooltip(me.tooltip, true);
         }
-        if(this.tooltip){
-            this.setTooltip(this.tooltip, true);
+
+        if (me.textAlign) {
+            me.setTextAlign(me.textAlign);
         }
 
-        if(this.handleMouseEvents){
-            this.mon(btn, {
-                scope: this,
-                mouseover: this.onMouseOver,
-                mousedown: this.onMouseDown
+        // Add the mouse events to the button
+        if (me.handleMouseEvents) {
+            me.mon(btn, {
+                scope: me,
+                mouseover: me.onMouseOver,
+                mouseout: me.onMouseOut,
+                mousedown: me.onMouseDown
             });
 
-            // new functionality for monitoring on the document level
-            //this.mon(btn, 'mouseout', this.onMouseOut, this);
+            if (me.split) {
+                me.mon(btn, {
+                    mousemove: me.onMouseMove,
+                    scope: me
+                });
+            }
         }
 
-        if(this.menu){
-            this.mon(this.menu, {
-                scope: this,
-                show: this.onMenuShow,
-                hide: this.onMenuHide
+        // Check if the button has a menu
+        if (me.menu) {
+            me.mon(me.menu, {
+                scope: me,
+                show: me.onMenuShow,
+                hide: me.onMenuHide
+            });
+
+            me.keyMap = Ext.create('Ext.util.KeyMap', me.el, {
+                key: Ext.EventObject.DOWN,
+                handler: me.onDownKey,
+                scope: me
             });
         }
 
-        if(this.repeat){
-            var repeater = new Ext.util.ClickRepeater(btn, Ext.isObject(this.repeat) ? this.repeat : {});
-            this.mon(repeater, 'click', this.onClick, this);
+        // Check if it is a repeat button
+        if (me.repeat) {
+            repeater = Ext.create('Ext.util.ClickRepeater', btn, Ext.isObject(me.repeat) ? me.repeat: {});
+            me.mon(repeater, 'click', me.onRepeatClick, me);
+        } else {
+            me.mon(btn, me.clickEvent, me.onClick, me);
         }
 
-        this.mon(btn, this.clickEvent, this.onClick, this);
+        // Register the button in the toggle manager
+        Ext.ButtonToggleManager.register(me);
+    },
+
+<span id='Ext-button-Button-method-getTemplateArgs'>    /**
+</span>     * This method returns an object which provides substitution parameters for the {@link #renderTpl XTemplate} used to
+     * create this Button's DOM structure.
+     *
+     * Instances or subclasses which use a different Template to create a different DOM structure may need to provide
+     * their own implementation of this method.
+     *
+     * @return {Object} Substitution data for a Template. The default implementation which provides data for the default
+     * {@link #template} returns an Object containing the following properties:
+     * @return {String} return.type The `&lt;button&gt;`'s {@link #type}
+     * @return {String} return.splitCls A CSS class to determine the presence and position of an arrow icon.
+     * (`'x-btn-arrow'` or `'x-btn-arrow-bottom'` or `''`)
+     * @return {String} return.cls A CSS class name applied to the Button's main `&lt;tbody&gt;` element which determines the
+     * button's scale and icon alignment.
+     * @return {String} return.text The {@link #text} to display ion the Button.
+     * @return {Number} return.tabIndex The tab index within the input flow.
+     */
+    getTemplateArgs: function() {
+        var me = this,
+            persistentPadding = me.getPersistentBtnPadding(),
+            innerSpanStyle = '';
+
+        // Create negative margin offsets to counteract persistent button padding if needed
+        if (Math.max.apply(Math, persistentPadding) &gt; 0) {
+            innerSpanStyle = 'margin:' + Ext.Array.map(persistentPadding, function(pad) {
+                return -pad + 'px';
+            }).join(' ');
+        }
+
+        return {
+            href     : me.getHref(),
+            target   : me.target || '_blank',
+            type     : me.type,
+            splitCls : me.getSplitCls(),
+            cls      : me.cls,
+            iconCls  : me.iconCls || '',
+            text     : me.text || '&amp;#160;',
+            tabIndex : me.tabIndex,
+            innerSpanStyle: innerSpanStyle
+        };
+    },
+
+<span id='Ext-button-Button-method-getHref'>    /**
+</span>     * @private
+     * If there is a configured href for this Button, returns the href with parameters appended.
+     * @returns The href string with parameters appended.
+     */
+    getHref: function() {
+        var me = this,
+            params = Ext.apply({}, me.baseParams);
+
+        // write baseParams first, then write any params
+        params = Ext.apply(params, me.params);
+        return me.href ? Ext.urlAppend(me.href, Ext.Object.toQueryString(params)) : false;
+    },
+
+<span id='Ext-button-Button-method-setParams'>    /**
+</span>     * Sets the href of the link dynamically according to the params passed, and any {@link #baseParams} configured.
+     *
+     * **Only valid if the Button was originally configured with a {@link #href}**
+     *
+     * @param {Object} params Parameters to use in the href URL.
+     */
+    setParams: function(params) {
+        this.params = params;
+        this.btnEl.dom.href = this.getHref();
+    },
+
+    getSplitCls: function() {
+        var me = this;
+        return me.split ? (me.baseCls + '-' + me.arrowCls) + ' ' + (me.baseCls + '-' + me.arrowCls + '-' + me.arrowAlign) : '';
     },
 
     // private
-    afterRender : function(){
-        Ext.Button.superclass.afterRender.call(this);
-        this.doAutoWidth();
+    afterRender: function() {
+        var me = this;
+        me.useSetClass = true;
+        me.setButtonCls();
+        me.doc = Ext.getDoc();
+        this.callParent(arguments);
     },
 
-    /**
-     * 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.
+<span id='Ext-button-Button-method-setIconCls'>    /**
+</span>     * 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
+     * @return {Ext.button.Button} this
      */
-    setIconClass : function(cls){
-        if(this.el){
-            this.btnEl.replaceClass(this.iconCls, cls);
+    setIconCls: function(cls) {
+        var me = this,
+            btnIconEl = me.btnIconEl,
+            oldCls = me.iconCls;
+            
+        me.iconCls = cls;
+        if (btnIconEl) {
+            // Remove the previous iconCls from the button
+            btnIconEl.removeCls(oldCls);
+            btnIconEl.addCls(cls || '');
+            me.setButtonCls();
         }
-        this.iconCls = cls;
-        return this;
+        return me;
     },
 
-    /**
-     * 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
+<span id='Ext-button-Button-method-setTooltip'>    /**
+</span>     * Sets the tooltip for this Button.
+     *
+     * @param {String/Object} tooltip This may be:
+     *
+     *   - **String** : A string to be used as innerHTML (html tags are accepted) to show in a tooltip
+     *   - **Object** : A configuration object for {@link Ext.tip.QuickTipManager#register}.
+     *
+     * @return {Ext.button.Button} this
      */
-    setTooltip : function(tooltip, /* private */ initial){
-        if(this.rendered){
-            if(!initial){
-                this.clearTip();
+    setTooltip: function(tooltip, initial) {
+        var me = this;
+
+        if (me.rendered) {
+            if (!initial) {
+                me.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;
+            if (Ext.isObject(tooltip)) {
+                Ext.tip.QuickTipManager.register(Ext.apply({
+                    target: me.btnEl.id
+                },
+                tooltip));
+                me.tooltip = tooltip;
+            } else {
+                me.btnEl.dom.setAttribute(me.getTipAttr(), tooltip);
             }
-        }else{
-            this.tooltip = tooltip;
+        } else {
+            me.tooltip = tooltip;
         }
-        return this;
+        return me;
+    },
+
+<span id='Ext-button-Button-method-setTextAlign'>    /**
+</span>     * Sets the text alignment for this button.
+     * @param {String} align The new alignment of the button text. See {@link #textAlign}.
+     */
+    setTextAlign: function(align) {
+        var me = this,
+            btnEl = me.btnEl;
+
+        if (btnEl) {
+            btnEl.removeCls(me.baseCls + '-' + me.textAlign);
+            btnEl.addCls(me.baseCls + '-' + align);
+        }
+        me.textAlign = align;
+        return me;
+    },
+
+    getTipAttr: function(){
+        return this.tooltipType == 'qtip' ? 'data-qtip' : 'title';
     },
 
     // private
-    clearTip : function(){
-        if(Ext.isObject(this.tooltip)){
-            Ext.QuickTips.unregister(this.btnEl);
+    getRefItems: function(deep){
+        var menu = this.menu,
+            items;
+        
+        if (menu) {
+            items = menu.getRefItems(deep);
+            items.unshift(menu);
         }
+        return items || [];
     },
 
     // private
-    beforeDestroy : function(){
-        if(this.rendered){
-            this.clearTip();
+    clearTip: function() {
+        if (Ext.isObject(this.tooltip)) {
+            Ext.tip.QuickTipManager.unregister(this.btnEl);
         }
-        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);
+    beforeDestroy: function() {
+        var me = this;
+        if (me.rendered) {
+            me.clearTip();
+        }
+        if (me.menu &amp;&amp; me.destroyMenu !== false) {
+            Ext.destroy(me.menu);
         }
+        Ext.destroy(me.btnInnerEl, me.repeater);
+        me.callParent();
     },
 
     // 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);
-                }
-            }
+    onDestroy: function() {
+        var me = this;
+        if (me.rendered) {
+            me.doc.un('mouseover', me.monitorMouseOver, me);
+            me.doc.un('mouseup', me.onMouseUp, me);
+            delete me.doc;
+            Ext.ButtonToggleManager.unregister(me);
+
+            Ext.destroy(me.keyMap);
+            delete me.keyMap;
         }
+        me.callParent();
     },
 
-    /**
-     * Assigns this Button's click handler
+<span id='Ext-button-Button-method-setHandler'>    /**
+</span>     * 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. Defaults to this Button.
-     * @return {Ext.Button} this
+     * @param {Object} [scope] The scope (`this` reference) in which the handler function is executed.
+     * Defaults to this Button.
+     * @return {Ext.button.Button} this
      */
-    setHandler : function(handler, scope){
+    setHandler: function(handler, scope) {
         this.handler = handler;
         this.scope = scope;
         return this;
     },
 
-    /**
-     * Sets this Button's text
+<span id='Ext-button-Button-method-setText'>    /**
+</span>     * Sets this Button's text
      * @param {String} text The button text
-     * @return {Ext.Button} this
+     * @return {Ext.button.Button} this
      */
-    setText : function(text){
-        this.text = text;
-        if(this.el){
-            this.el.child('td.x-btn-mc ' + this.buttonSelector).update(text);
+    setText: function(text) {
+        var me = this;
+        me.text = text;
+        if (me.el) {
+            me.btnInnerEl.update(text || '&amp;#160;');
+            me.setButtonCls();
         }
-        this.doAutoWidth();
-        return this;
+        me.doComponentLayout();
+        return me;
+    },
+
+<span id='Ext-button-Button-method-setIcon'>    /**
+</span>     * Sets the background image (inline style) of the button. This method also changes the value of the {@link #icon}
+     * config internally.
+     * @param {String} icon The path to an image to display in the button
+     * @return {Ext.button.Button} this
+     */
+    setIcon: function(icon) {
+        var me = this,
+            iconEl = me.btnIconEl;
+            
+        me.icon = icon;
+        if (iconEl) {
+            iconEl.setStyle('background-image', icon ? 'url(' + icon + ')': '');
+            me.setButtonCls();
+        }
+        return me;
     },
 
-    /**
-     * Gets the text for this Button
+<span id='Ext-button-Button-method-getText'>    /**
+</span>     * Gets the text for this Button
      * @return {String} The button text
      */
-    getText : function(){
+    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
+<span id='Ext-button-Button-method-toggle'>    /**
+</span>     * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
+     * @param {Boolean} [state] Force a particular state
+     * @param {Boolean} [suppressEvent=false] True to stop events being fired when calling this method.
+     * @return {Ext.button.Button} this
      */
-    toggle : function(state, suppressEvent){
-        state = state === undefined ? !this.pressed : !!state;
-        if(state != this.pressed){
-            if(this.rendered){
-                this.el[state ? 'addClass' : 'removeClass']('x-btn-pressed');
+    toggle: function(state, suppressEvent) {
+        var me = this;
+        state = state === undefined ? !me.pressed : !!state;
+        if (state !== me.pressed) {
+            if (me.rendered) {
+                me[state ? 'addClsWithUI': 'removeClsWithUI'](me.pressedCls);
             }
-            this.pressed = state;
-            if(!suppressEvent){
-                this.fireEvent('toggle', this, state);
-                if(this.toggleHandler){
-                    this.toggleHandler.call(this.scope || this, this, state);
-                }
+            me.btnEl.dom.setAttribute('aria-pressed', state);
+            me.pressed = state;
+            if (!suppressEvent) {
+                me.fireEvent('toggle', me, state);
+                Ext.callback(me.toggleHandler, me.scope || me, [me, state]);
             }
         }
-        return this;
-    },
-
-    /**
-     * Focus the button
-     */
-    focus : function(){
-        this.btnEl.focus();
+        return me;
     },
-
-    // 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;
+    
+    maybeShowMenu: function(){
+        var me = this;
+        if (me.menu &amp;&amp; !me.hasVisibleMenu() &amp;&amp; !me.ignoreNextClick) {
+            me.showMenu();
         }
-        this.disabled = disabled;
     },
 
-    /**
-     * Show this button's menu (if it has one)
+<span id='Ext-button-Button-method-showMenu'>    /**
+</span>     * Shows this button's menu (if it has one)
      */
-    showMenu : function(){
-        if(this.rendered && this.menu){
-            if(this.tooltip){
-                Ext.QuickTips.getQuickTip().cancelShow(this.btnEl);
+    showMenu: function() {
+        var me = this;
+        if (me.rendered &amp;&amp; me.menu) {
+            if (me.tooltip &amp;&amp; me.getTipAttr() != 'title') {
+                Ext.tip.QuickTipManager.getQuickTip().cancelShow(me.btnEl);
             }
-            this.menu.show(this.el, this.menuAlign);
+            if (me.menu.isVisible()) {
+                me.menu.hide();
+            }
+
+            me.menu.showBy(me.el, me.menuAlign);
         }
-        return this;
+        return me;
     },
 
-    /**
-     * Hide this button's menu (if it has one)
+<span id='Ext-button-Button-method-hideMenu'>    /**
+</span>     * Hides this button's menu (if it has one)
      */
-    hideMenu : function(){
-        if(this.menu){
+    hideMenu: function() {
+        if (this.hasVisibleMenu()) {
             this.menu.hide();
         }
         return this;
     },
 
-    /**
-     * Returns true if the button has a menu and it is visible
+<span id='Ext-button-Button-method-hasVisibleMenu'>    /**
+</span>     * Returns true if the button has a menu and it is visible
      * @return {Boolean}
      */
-    hasVisibleMenu : function(){
-        return this.menu && this.menu.isVisible();
+    hasVisibleMenu: function() {
+        var menu = this.menu;
+        return menu &amp;&amp; menu.rendered &amp;&amp; menu.isVisible();
+    },
+
+    // private
+    onRepeatClick: function(repeat, e) {
+        this.onClick(e);
     },
 
     // private
-    onClick : function(e){
-        if(e){
+    onClick: function(e) {
+        var me = this;
+        if (me.preventDefault || (me.disabled &amp;&amp; me.getHref()) &amp;&amp; e) {
             e.preventDefault();
         }
-        if(e.button !== 0){
+        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);
-            }
+        if (!me.disabled) {
+            me.doToggle();
+            me.maybeShowMenu();
+            me.fireHandler(e);
+        }
+    },
+    
+    fireHandler: function(e){
+        var me = this,
+            handler = me.handler;
+            
+        me.fireEvent('click', me, e);
+        if (handler) {
+            handler.call(me.scope || me, me, e);
+        }
+        me.onBlur();
+    },
+    
+    doToggle: function(){
+        var me = this;
+        if (me.enableToggle &amp;&amp; (me.allowDepress !== false || !me.pressed)) {
+            me.toggle();
         }
     },
 
-    // private
-    isMenuTriggerOver : function(e, internal){
-        return this.menu && !internal;
+<span id='Ext-button-Button-method-onMouseOver'>    /**
+</span>     * @private mouseover handler called when a mouseover event occurs anywhere within the encapsulating element.
+     * The targets are interrogated to see what is being entered from where.
+     * @param e
+     */
+    onMouseOver: function(e) {
+        var me = this;
+        if (!me.disabled &amp;&amp; !e.within(me.el, true, true)) {
+            me.onMouseEnter(e);
+        }
     },
 
-    // private
-    isMenuTriggerOut : function(e, internal){
-        return this.menu && !internal;
+<span id='Ext-button-Button-method-onMouseOut'>    /**
+</span>     * @private
+     * mouseout handler called when a mouseout event occurs anywhere within the encapsulating element -
+     * or the mouse leaves the encapsulating element.
+     * The targets are interrogated to see what is being exited to where.
+     * @param e
+     */
+    onMouseOut: function(e) {
+        var me = this;
+        if (!e.within(me.el, true, true)) {
+            if (me.overMenuTrigger) {
+                me.onMenuTriggerOut(e);
+            }
+            me.onMouseLeave(e);
+        }
     },
 
-    // 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);
+<span id='Ext-button-Button-method-onMouseMove'>    /**
+</span>     * @private
+     * mousemove handler called when the mouse moves anywhere within the encapsulating element.
+     * The position is checked to determine if the mouse is entering or leaving the trigger area. Using
+     * mousemove to check this is more resource intensive than we'd like, but it is necessary because
+     * the trigger area does not line up exactly with sub-elements so we don't always get mouseover/out
+     * events when needed. In the future we should consider making the trigger a separate element that
+     * is absolutely positioned and sized over the trigger area.
+     */
+    onMouseMove: function(e) {
+        var me = this,
+            el = me.el,
+            over = me.overMenuTrigger,
+            overlap, btnSize;
+
+        if (me.split) {
+            if (me.arrowAlign === 'right') {
+                overlap = e.getX() - el.getX();
+                btnSize = el.getWidth();
+            } else {
+                overlap = e.getY() - el.getY();
+                btnSize = el.getHeight();
             }
-            if(this.isMenuTriggerOver(e, internal)){
-                this.fireEvent('menutriggerover', this, this.menu, e);
+
+            if (overlap &gt; (btnSize - me.getTriggerSize())) {
+                if (!over) {
+                    me.onMenuTriggerOver(e);
+                }
+            } else {
+                if (over) {
+                    me.onMenuTriggerOut(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);
+<span id='Ext-button-Button-method-getTriggerSize'>    /**
+</span>     * @private
+     * Measures the size of the trigger area for menu and split buttons. Will be a width for
+     * a right-aligned trigger and a height for a bottom-aligned trigger. Cached after first measurement.
+     */
+    getTriggerSize: function() {
+        var me = this,
+            size = me.triggerSize,
+            side, sideFirstLetter, undef;
+
+        if (size === undef) {
+            side = me.arrowAlign;
+            sideFirstLetter = side.charAt(0);
+            size = me.triggerSize = me.el.getFrameWidth(sideFirstLetter) + me.btnWrap.getFrameWidth(sideFirstLetter) + (me.frameSize &amp;&amp; me.frameSize[side] || 0);
         }
+        return size;
     },
 
-    // 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);
+<span id='Ext-button-Button-method-onMouseEnter'>    /**
+</span>     * @private
+     * virtual mouseenter handler called when it is detected that the mouseout event
+     * signified the mouse entering the encapsulating element.
+     * @param e
+     */
+    onMouseEnter: function(e) {
+        var me = this;
+        me.addClsWithUI(me.overCls);
+        me.fireEvent('mouseover', me, e);
+    },
+
+<span id='Ext-button-Button-method-onMouseLeave'>    /**
+</span>     * @private
+     * virtual mouseleave handler called when it is detected that the mouseover event
+     * signified the mouse entering the encapsulating element.
+     * @param e
+     */
+    onMouseLeave: function(e) {
+        var me = this;
+        me.removeClsWithUI(me.overCls);
+        me.fireEvent('mouseout', me, e);
+    },
+
+<span id='Ext-button-Button-method-onMenuTriggerOver'>    /**
+</span>     * @private
+     * virtual mouseenter handler called when it is detected that the mouseover event
+     * signified the mouse entering the arrow area of the button - the &lt;em&gt;.
+     * @param e
+     */
+    onMenuTriggerOver: function(e) {
+        var me = this;
+        me.overMenuTrigger = true;
+        me.fireEvent('menutriggerover', me, me.menu, e);
+    },
+
+<span id='Ext-button-Button-method-onMenuTriggerOut'>    /**
+</span>     * @private
+     * virtual mouseleave handler called when it is detected that the mouseout event
+     * signified the mouse leaving the arrow area of the button - the &lt;em&gt;.
+     * @param e
+     */
+    onMenuTriggerOut: function(e) {
+        var me = this;
+        delete me.overMenuTrigger;
+        me.fireEvent('menutriggerout', me, me.menu, e);
+    },
+
+    // inherit docs
+    enable : function(silent) {
+        var me = this;
+
+        me.callParent(arguments);
+
+        me.removeClsWithUI('disabled');
+
+        return me;
+    },
+
+    // inherit docs
+    disable : function(silent) {
+        var me = this;
+
+        me.callParent(arguments);
+
+        me.addClsWithUI('disabled');
+        me.removeClsWithUI(me.overCls);
+
+        return me;
+    },
+
+<span id='Ext-button-Button-method-setScale'>    /**
+</span>     * Method to change the scale of the button. See {@link #scale} for allowed configurations.
+     * @param {String} scale The scale to change to.
+     */
+    setScale: function(scale) {
+        var me = this,
+            ui = me.ui.replace('-' + me.scale, '');
+
+        //check if it is an allowed scale
+        if (!Ext.Array.contains(me.allowedScales, scale)) {
+            throw('#setScale: scale must be an allowed scale (' + me.allowedScales.join(', ') + ')');
         }
+
+        me.scale = scale;
+        me.setUI(ui);
     },
-    // private
-    onFocus : function(e){
-        if(!this.disabled){
-            this.el.addClass('x-btn-focus');
+
+    // inherit docs
+    setUI: function(ui) {
+        var me = this;
+
+        //we need to append the scale to the UI, if not already done
+        if (me.scale &amp;&amp; !ui.match(me.scale)) {
+            ui = ui + '-' + me.scale;
         }
+
+        me.callParent([ui]);
+
+        // Set all the state classNames, as they need to include the UI
+        // me.disabledCls += ' ' + me.baseCls + '-' + me.ui + '-disabled';
     },
+
     // private
-    onBlur : function(e){
-        this.el.removeClass('x-btn-focus');
+    onFocus: function(e) {
+        var me = this;
+        if (!me.disabled) {
+            me.addClsWithUI(me.focusCls);
+        }
     },
 
     // private
-    getClickEl : function(e, isUp){
-       return this.el;
+    onBlur: function(e) {
+        var me = this;
+        me.removeClsWithUI(me.focusCls);
     },
 
     // private
-    onMouseDown : function(e){
-        if(!this.disabled && e.button === 0){
-            this.getClickEl(e).addClass('x-btn-click');
-            Ext.getDoc().on('mouseup', this.onMouseUp, this);
+    onMouseDown: function(e) {
+        var me = this;
+        if (!me.disabled &amp;&amp; e.button === 0) {
+            me.addClsWithUI(me.pressedCls);
+            me.doc.on('mouseup', me.onMouseUp, me);
         }
     },
     // private
-    onMouseUp : function(e){
-        if(e.button === 0){
-            this.getClickEl(e, true).removeClass('x-btn-click');
-            Ext.getDoc().un('mouseup', this.onMouseUp, this);
+    onMouseUp: function(e) {
+        var me = this;
+        if (e.button === 0) {
+            if (!me.pressed) {
+                me.removeClsWithUI(me.pressedCls);
+            }
+            me.doc.un('mouseup', me.onMouseUp, me);
         }
     },
     // private
-    onMenuShow : function(e){
-        this.ignoreNextClick = 0;
-        this.el.addClass('x-btn-menu-active');
-        this.fireEvent('menushow', this, this.menu);
+    onMenuShow: function(e) {
+        var me = this;
+        me.ignoreNextClick = 0;
+        me.addClsWithUI(me.menuActiveCls);
+        me.fireEvent('menushow', me, me.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);
+    onMenuHide: function(e) {
+        var me = this;
+        me.removeClsWithUI(me.menuActiveCls);
+        me.ignoreNextClick = Ext.defer(me.restoreClick, 250, me);
+        me.fireEvent('menuhide', me, me.menu);
     },
 
     // private
-    restoreClick : function(){
+    restoreClick: function() {
         this.ignoreNextClick = 0;
+    },
+
+    // private
+    onDownKey: function() {
+        var me = this;
+
+        if (!me.disabled) {
+            if (me.menu) {
+                me.showMenu();
+            }
+        }
+    },
+
+<span id='Ext-button-Button-method-getPersistentBtnPadding'>    /**
+</span>     * @private
+     * Some browsers (notably Safari and older Chromes on Windows) add extra &quot;padding&quot; inside the button
+     * element that cannot be removed. This method returns the size of that padding with a one-time detection.
+     * @return {Number[]} [top, right, bottom, left]
+     */
+    getPersistentBtnPadding: function() {
+        var cls = Ext.button.Button,
+            padding = cls.persistentPadding,
+            btn, leftTop, btnEl, btnInnerEl;
+
+        if (!padding) {
+            padding = cls.persistentPadding = [0, 0, 0, 0]; //set early to prevent recursion
+
+            if (!Ext.isIE) { //short-circuit IE as it sometimes gives false positive for padding
+                // Create auto-size button offscreen and measure its insides
+                btn = Ext.create('Ext.button.Button', {
+                    renderTo: Ext.getBody(),
+                    text: 'test',
+                    style: 'position:absolute;top:-999px;'
+                });
+                btnEl = btn.btnEl;
+                btnInnerEl = btn.btnInnerEl;
+                btnEl.setSize(null, null); //clear any hard dimensions on the button el to see what it does naturally
+
+                leftTop = btnInnerEl.getOffsetsTo(btnEl);
+                padding[0] = leftTop[1];
+                padding[1] = btnEl.getWidth() - btnInnerEl.getWidth() - leftTop[0];
+                padding[2] = btnEl.getHeight() - btnInnerEl.getHeight() - leftTop[1];
+                padding[3] = leftTop[0];
+
+                btn.destroy();
+            }
+        }
+
+        return padding;
     }
 
+}, function() {
+    var groups = {};
 
+    function toggleGroup(btn, state) {
+        var g, i, l;
+        if (state) {
+            g = groups[btn.toggleGroup];
+            for (i = 0, l = g.length; i &lt; l; i++) {
+                if (g[i] !== btn) {
+                    g[i].toggle(false);
+                }
+            }
+        }
+    }
 
-    /**
-     * @cfg {String} autoEl @hide
+<span id='Ext-ButtonToggleManager'>    /**
+</span>     * Private utility class used by Button
+     * @hide
      */
+    Ext.ButtonToggleManager = {
+        register: function(btn) {
+            if (!btn.toggleGroup) {
+                return;
+            }
+            var group = groups[btn.toggleGroup];
+            if (!group) {
+                group = groups[btn.toggleGroup] = [];
+            }
+            group.push(btn);
+            btn.on('toggle', toggleGroup);
+        },
+
+        unregister: function(btn) {
+            if (!btn.toggleGroup) {
+                return;
+            }
+            var group = groups[btn.toggleGroup];
+            if (group) {
+                Ext.Array.remove(group, btn);
+                btn.un('toggle', toggleGroup);
+            }
+        },
+
+<span id='Ext-ButtonToggleManager-method-getPressed'>        /**
+</span>         * Gets the pressed button in the passed group or null
+         * @param {String} group
+         * @return {Ext.button.Button}
+         */
+        getPressed: function(group) {
+            var g = groups[group],
+                i = 0,
+                len;
+            if (g) {
+                for (len = g.length; i &lt; len; i++) {
+                    if (g[i].pressed === true) {
+                        return g[i];
+                    }
+                }
+            }
+            return null;
+        }
+    };
 });
-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;
-       }
-   };
-}();</pre>    \r
-</body>\r
-</html>
\ No newline at end of file
+</pre>
+</body>
+</html>