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

中间件编程

开发平台:

JavaScript

  1. /*!  * Ext JS Library 3.0.0  * Copyright(c) 2006-2009 Ext JS, LLC  * licensing@extjs.com  * http://www.extjs.com/license  */ /**
  2.  * @class Ext.layout.MenuLayout
  3.  * @extends Ext.layout.ContainerLayout
  4.  * <p>Layout manager used by {@link Ext.menu.Menu}. Generally this class should not need to be used directly.</p>
  5.  */
  6.  Ext.layout.MenuLayout = Ext.extend(Ext.layout.ContainerLayout, {
  7.     monitorResize: true,
  8.     setContainer : function(ct){
  9.         this.monitorResize = !ct.floating;
  10.         Ext.layout.MenuLayout.superclass.setContainer.call(this, ct);
  11.     },
  12.     renderItem : function(c, position, target){
  13.         if (!this.itemTpl) {
  14.             this.itemTpl = Ext.layout.MenuLayout.prototype.itemTpl = new Ext.XTemplate(
  15.                 '<li id="{itemId}" class="{itemCls}">',
  16.                     '<tpl if="needsIcon">',
  17.                         '<img src="{icon}" class="{iconCls}"/>',
  18.                     '</tpl>',
  19.                 '</li>'
  20.             );
  21.         }
  22.         if(c && !c.rendered){
  23.             if(Ext.isNumber(position)){
  24.                 position = target.dom.childNodes[position];
  25.             }
  26.             var a = this.getItemArgs(c);
  27. //          The Component's positionEl is the <li> it is rendered into
  28.             c.render(c.positionEl = position ?
  29.                 this.itemTpl.insertBefore(position, a, true) :
  30.                 this.itemTpl.append(target, a, true));
  31. //          Link the containing <li> to the item.
  32.             c.positionEl.menuItemId = c.itemId || c.id;
  33. //          If rendering a regular Component, and it needs an icon,
  34. //          move the Component rightwards.
  35.             if (!a.isMenuItem && a.needsIcon) {
  36.                 c.positionEl.addClass('x-menu-list-item-indent');
  37.             }
  38.         }else if(c && !this.isValidParent(c, target)){
  39.             if(Ext.isNumber(position)){
  40.                 position = target.dom.childNodes[position];
  41.             }
  42.             target.dom.insertBefore(c.getActionEl().dom, position || null);
  43.         }
  44.     },
  45.     getItemArgs: function(c) {
  46.         var isMenuItem = c instanceof Ext.menu.Item;
  47.         return {
  48.             isMenuItem: isMenuItem,
  49.             needsIcon: !isMenuItem && (c.icon || c.iconCls),
  50.             icon: c.icon || Ext.BLANK_IMAGE_URL,
  51.             iconCls: 'x-menu-item-icon ' + (c.iconCls || ''),
  52.             itemId: 'x-menu-el-' + c.id,
  53.             itemCls: 'x-menu-list-item ' + (this.extraCls || '')
  54.         };
  55.     },
  56. //  Valid if the Component is in a <li> which is part of our target <ul>
  57.     isValidParent: function(c, target) {
  58.         return c.el.up('li.x-menu-list-item', 5).dom.parentNode === (target.dom || target);
  59.     },
  60.     onLayout : function(ct, target){
  61.         this.renderAll(ct, target);
  62.         this.doAutoSize();
  63.     },
  64.     doAutoSize : function(){
  65.         var ct = this.container, w = ct.width;
  66.         if(ct.floating){
  67.             if(w){
  68.                 ct.setWidth(w);
  69.             }else if(Ext.isIE){
  70.                 ct.setWidth(Ext.isStrict && (Ext.isIE7 || Ext.isIE8) ? 'auto' : ct.minWidth);
  71.                 var el = ct.getEl(), t = el.dom.offsetWidth; // force recalc
  72.                 ct.setWidth(ct.getLayoutTarget().getWidth() + el.getFrameWidth('lr'));
  73.             }
  74.         }
  75.     }
  76. });
  77. Ext.Container.LAYOUTS['menu'] = Ext.layout.MenuLayout;
  78. /**
  79.  * @class Ext.menu.Menu
  80.  * @extends Ext.Container
  81.  * <p>A menu object.  This is the container to which you may add menu items.  Menu can also serve as a base class
  82.  * when you want a specialized menu based off of another component (like {@link Ext.menu.DateMenu} for example).</p>
  83.  * <p>Menus may contain either {@link Ext.menu.Item menu items}, or general {@link Ext.Component Component}s.</p>
  84.  * <p>To make a contained general {@link Ext.Component Component} line up with other {@link Ext.menu.Item menu items}
  85.  * specify <tt>iconCls: 'no-icon'</tt>.  This reserves a space for an icon, and indents the Component in line
  86.  * with the other menu items.  See {@link Ext.form.ComboBox}.{@link Ext.form.ComboBox#getListParent getListParent}
  87.  * for an example.</p>
  88.  * <p>By default, Menus are absolutely positioned, floating Components. By configuring a Menu with
  89.  * <b><tt>{@link #floating}:false</tt></b>, a Menu may be used as child of a Container.</p>
  90.  *
  91.  * @xtype menu
  92.  */
  93. Ext.menu.Menu = Ext.extend(Ext.Container, {
  94.     /**
  95.      * @cfg {Object} defaults
  96.      * A config object that will be applied to all items added to this container either via the {@link #items}
  97.      * config or via the {@link #add} method.  The defaults config can contain any number of
  98.      * name/value property pairs to be added to each item, and should be valid for the types of items
  99.      * being added to the menu.
  100.      */
  101.     /**
  102.      * @cfg {Mixed} items
  103.      * An array of items to be added to this menu. Menus may contain either {@link Ext.menu.Item menu items},
  104.      * or general {@link Ext.Component Component}s.
  105.      */
  106.     /**
  107.      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
  108.      */
  109.     minWidth : 120,
  110.     /**
  111.      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
  112.      * for bottom-right shadow (defaults to "sides")
  113.      */
  114.     shadow : "sides",
  115.     /**
  116.      * @cfg {String} subMenuAlign The {@link Ext.Element#alignTo} anchor position value to use for submenus of
  117.      * this menu (defaults to "tl-tr?")
  118.      */
  119.     subMenuAlign : "tl-tr?",
  120.     /**
  121.      * @cfg {String} defaultAlign The default {@link Ext.Element#alignTo} anchor position value for this menu
  122.      * relative to its element of origin (defaults to "tl-bl?")
  123.      */
  124.     defaultAlign : "tl-bl?",
  125.     /**
  126.      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
  127.      */
  128.     allowOtherMenus : false,
  129.     /**
  130.      * @cfg {Boolean} ignoreParentClicks True to ignore clicks on any item in this menu that is a parent item (displays
  131.      * a submenu) so that the submenu is not dismissed when clicking the parent item (defaults to false).
  132.      */
  133.     ignoreParentClicks : false,
  134.     /**
  135.      * @cfg {Boolean} enableScrolling True to allow the menu container to have scroller controls if the menu is too long (defaults to true).
  136.      */
  137.     enableScrolling: true,
  138.     /**
  139.      * @cfg {Number} maxHeight The maximum height of the menu. Only applies when enableScrolling is set to True (defaults to null).
  140.      */
  141.     maxHeight: null,
  142.     /**
  143.      * @cfg {Number} scrollIncrement The amount to scroll the menu. Only applies when enableScrolling is set to True (defaults to 24).
  144.      */
  145.     scrollIncrement: 24,
  146.     /**
  147.      * @cfg {Boolean} showSeparator True to show the icon separator. (defaults to true).
  148.      */
  149.     showSeparator: true,
  150.     /**
  151.      * @cfg {Array} defaultOffsets An array specifying the [x, y] offset in pixels by which to
  152.      * change the default Menu popup position after aligning according to the {@link #defaultAlign}
  153.      * configuration. Defaults to <tt>[0, 0]</tt>.
  154.      */
  155.     defaultOffsets : [0, 0],
  156.     /**
  157.      * @cfg {Boolean} floating
  158.      * May be specified as false to create a Menu which may be used as a child item of another Container
  159.      * instead of a free-floating {@link Ext.Layer Layer}. (defaults to true).
  160.      */
  161.     floating: true,         // Render as a Layer by default
  162.     // private
  163.     hidden: true,
  164.     layout: 'menu',
  165.     hideMode: 'offsets',    // Important for laying out Components
  166.     scrollerHeight: 8,
  167.     autoLayout: true,       // Provided for backwards compat
  168.     defaultType: 'menuitem',
  169.     initComponent: function(){
  170.         if(Ext.isArray(this.initialConfig)){
  171.             Ext.apply(this, {items:this.initialConfig});
  172.         }
  173.         this.addEvents(
  174.             /**
  175.              * @event click
  176.              * Fires when this menu is clicked (or when the enter key is pressed while it is active)
  177.              * @param {Ext.menu.Menu} this
  178.             * @param {Ext.menu.Item} menuItem The menu item that was clicked
  179.              * @param {Ext.EventObject} e
  180.              */
  181.             'click',
  182.             /**
  183.              * @event mouseover
  184.              * Fires when the mouse is hovering over this menu
  185.              * @param {Ext.menu.Menu} this
  186.              * @param {Ext.EventObject} e
  187.              * @param {Ext.menu.Item} menuItem The menu item that was clicked
  188.              */
  189.             'mouseover',
  190.             /**
  191.              * @event mouseout
  192.              * Fires when the mouse exits this menu
  193.              * @param {Ext.menu.Menu} this
  194.              * @param {Ext.EventObject} e
  195.              * @param {Ext.menu.Item} menuItem The menu item that was clicked
  196.              */
  197.             'mouseout',
  198.             /**
  199.              * @event itemclick
  200.              * Fires when a menu item contained in this menu is clicked
  201.              * @param {Ext.menu.BaseItem} baseItem The BaseItem that was clicked
  202.              * @param {Ext.EventObject} e
  203.              */
  204.             'itemclick'
  205.         );
  206.         Ext.menu.MenuMgr.register(this);
  207.         if(this.floating){
  208.             Ext.EventManager.onWindowResize(this.hide, this);
  209.         }else{
  210.             if(this.initialConfig.hidden !== false){
  211.                 this.hidden = false;
  212.             }
  213.             this.internalDefaults = {hideOnClick: false};
  214.         }
  215.         Ext.menu.Menu.superclass.initComponent.call(this);
  216.         if(this.autoLayout){
  217.             this.on({
  218.                 add: this.doLayout,
  219.                 remove: this.doLayout,
  220.                 scope: this
  221.             });
  222.         }
  223.     },
  224.     //private
  225.     getLayoutTarget : function() {
  226.         return this.ul;
  227.     },
  228.     // private
  229.     onRender : function(ct, position){
  230.         if(!ct){
  231.             ct = Ext.getBody();
  232.         }
  233.         var dh = {
  234.             id: this.getId(),
  235.             cls: 'x-menu ' + ((this.floating) ? 'x-menu-floating x-layer ' : '') + (this.cls || '') + (this.plain ? ' x-menu-plain' : '') + (this.showSeparator ? '' : ' x-menu-nosep'),
  236.             style: this.style,
  237.             cn: [
  238.                 {tag: 'a', cls: 'x-menu-focus', href: '#', onclick: 'return false;', tabIndex: '-1'},
  239.                 {tag: 'ul', cls: 'x-menu-list'}
  240.             ]
  241.         };
  242.         if(this.floating){
  243.             this.el = new Ext.Layer({
  244.                 shadow: this.shadow,
  245.                 dh: dh,
  246.                 constrain: false,
  247.                 parentEl: ct,
  248.                 zindex:15000
  249.             });
  250.         }else{
  251.             this.el = ct.createChild(dh);
  252.         }
  253.         Ext.menu.Menu.superclass.onRender.call(this, ct, position);
  254.         if(!this.keyNav){
  255.             this.keyNav = new Ext.menu.MenuNav(this);
  256.         }
  257.         // generic focus element
  258.         this.focusEl = this.el.child('a.x-menu-focus');
  259.         this.ul = this.el.child('ul.x-menu-list');
  260.         this.mon(this.ul, {
  261.             scope: this,
  262.             click: this.onClick,
  263.             mouseover: this.onMouseOver,
  264.             mouseout: this.onMouseOut
  265.         });
  266.         if(this.enableScrolling){
  267.             this.mon(this.el, {
  268.                 scope: this,
  269.                 delegate: '.x-menu-scroller',
  270.                 click: this.onScroll,
  271.                 mouseover: this.deactivateActive
  272.             });
  273.         }
  274.     },
  275.     // private
  276.     findTargetItem : function(e){
  277.         var t = e.getTarget(".x-menu-list-item", this.ul, true);
  278.         if(t && t.menuItemId){
  279.             return this.items.get(t.menuItemId);
  280.         }
  281.     },
  282.     // private
  283.     onClick : function(e){
  284.         var t = this.findTargetItem(e);
  285.         if(t){
  286.             if(t.isFormField){
  287.                 this.setActiveItem(t);
  288.             }else{
  289.                 if(t.menu && this.ignoreParentClicks){
  290.                     t.expandMenu();
  291.                     e.preventDefault();
  292.                 }else if(t.onClick){
  293.                     t.onClick(e);
  294.                     this.fireEvent("click", this, t, e);
  295.                 }
  296.             }
  297.         }
  298.     },
  299.     // private
  300.     setActiveItem : function(item, autoExpand){
  301.         if(item != this.activeItem){
  302.             this.deactivateActive();
  303.             if((this.activeItem = item).isFormField){
  304.                 item.focus();
  305.             }else{
  306.                 item.activate(autoExpand);
  307.             }
  308.         }else if(autoExpand){
  309.             item.expandMenu();
  310.         }
  311.     },
  312.     deactivateActive: function(){
  313.         var a = this.activeItem;
  314.         if(a){
  315.             if(a.isFormField){
  316.                 //Fields cannot deactivate, but Combos must collapse
  317.                 if(a.collapse){
  318.                     a.collapse();
  319.                 }
  320.             }else{
  321.                 a.deactivate();
  322.             }
  323.             delete this.activeItem;
  324.         }
  325.     },
  326.     // private
  327.     tryActivate : function(start, step){
  328.         var items = this.items;
  329.         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
  330.             var item = items.get(i);
  331.             if(!item.disabled && (item.canActivate || item.isFormField)){
  332.                 this.setActiveItem(item, false);
  333.                 return item;
  334.             }
  335.         }
  336.         return false;
  337.     },
  338.     // private
  339.     onMouseOver : function(e){
  340.         var t = this.findTargetItem(e);
  341.         if(t){
  342.             if(t.canActivate && !t.disabled){
  343.                 this.setActiveItem(t, true);
  344.             }
  345.         }
  346.         this.over = true;
  347.         this.fireEvent("mouseover", this, e, t);
  348.     },
  349.     // private
  350.     onMouseOut : function(e){
  351.         var t = this.findTargetItem(e);
  352.         if(t){
  353.             if(t == this.activeItem && t.shouldDeactivate && t.shouldDeactivate(e)){
  354.                 this.activeItem.deactivate();
  355.                 delete this.activeItem;
  356.             }
  357.         }
  358.         this.over = false;
  359.         this.fireEvent("mouseout", this, e, t);
  360.     },
  361.     // private
  362.     onScroll: function(e, t){
  363.         if(e){
  364.             e.stopEvent();
  365.         }
  366.         var ul = this.ul.dom, top = Ext.fly(t).is('.x-menu-scroller-top');
  367.         ul.scrollTop += this.scrollIncrement * (top ? -1 : 1);
  368.         if(top ? ul.scrollTop <= 0 : ul.scrollTop + this.activeMax >= ul.scrollHeight){
  369.            this.onScrollerOut(null, t);
  370.         }
  371.     },
  372.     // private
  373.     onScrollerIn: function(e, t){
  374.         var ul = this.ul.dom, top = Ext.fly(t).is('.x-menu-scroller-top');
  375.         if(top ? ul.scrollTop > 0 : ul.scrollTop + this.activeMax < ul.scrollHeight){
  376.             Ext.fly(t).addClass(['x-menu-item-active', 'x-menu-scroller-active']);
  377.         }
  378.     },
  379.     // private
  380.     onScrollerOut: function(e, t){
  381.         Ext.fly(t).removeClass(['x-menu-item-active', 'x-menu-scroller-active']);
  382.     },
  383.     /**
  384.      * Displays this menu relative to another element
  385.      * @param {Mixed} element The element to align to
  386.      * @param {String} position (optional) The {@link Ext.Element#alignTo} anchor position to use in aligning to
  387.      * the element (defaults to this.defaultAlign)
  388.      * @param {Ext.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
  389.      */
  390.     show : function(el, pos, parentMenu){
  391.         if(this.floating){
  392.             this.parentMenu = parentMenu;
  393.             if(!this.el){
  394.                 this.render();
  395.                 this.doLayout(false, true);
  396.             }
  397.             if(this.fireEvent('beforeshow', this) !== false){
  398.                 this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign, this.defaultOffsets), parentMenu, false);
  399.             }
  400.         }else{
  401.             Ext.menu.Menu.superclass.show.call(this);
  402.         }
  403.     },
  404.     /**
  405.      * Displays this menu at a specific xy position
  406.      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
  407.      * @param {Ext.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
  408.      */
  409.     showAt : function(xy, parentMenu, /* private: */_e){
  410.         this.parentMenu = parentMenu;
  411.         if(!this.el){
  412.             this.render();
  413.         }
  414.         this.el.setXY(xy);
  415.         if(this.enableScrolling){
  416.             this.constrainScroll(xy[1]);
  417.         }
  418.         this.el.show();
  419.         Ext.menu.Menu.superclass.onShow.call(this);
  420.         if(Ext.isIE){
  421.             this.layout.doAutoSize();
  422.             if(!Ext.isIE8){
  423.                 this.el.repaint();
  424.             }
  425.         }
  426.         this.hidden = false;
  427.         this.focus();
  428.         this.fireEvent("show", this);
  429.     },
  430.     constrainScroll: function(y){
  431.         var max, full = this.ul.setHeight('auto').getHeight();
  432.         if(this.floating){
  433.             max = this.maxHeight ? this.maxHeight : Ext.fly(this.el.dom.parentNode).getViewSize().height - y;
  434.         }else{
  435.             max = this.getHeight();
  436.         }
  437.         if(full > max && max > 0){
  438.             this.activeMax = max - this.scrollerHeight * 2 - this.el.getFrameWidth('tb') - Ext.num(this.el.shadowOffset, 0);
  439.             this.ul.setHeight(this.activeMax);
  440.             this.createScrollers();
  441.             this.el.select('.x-menu-scroller').setDisplayed('');
  442.         }else{
  443.             this.ul.setHeight(full);
  444.             this.el.select('.x-menu-scroller').setDisplayed('none');
  445.         }
  446.         this.ul.dom.scrollTop = 0;
  447.     },
  448.     createScrollers: function(){
  449.         if(!this.scroller){
  450.             this.scroller = {
  451.                 pos: 0,
  452.                 top: this.el.insertFirst({
  453.                     tag: 'div',
  454.                     cls: 'x-menu-scroller x-menu-scroller-top',
  455.                     html: '&#160;'
  456.                 }),
  457.                 bottom: this.el.createChild({
  458.                     tag: 'div',
  459.                     cls: 'x-menu-scroller x-menu-scroller-bottom',
  460.                     html: '&#160;'
  461.                 })
  462.             };
  463.             this.scroller.top.hover(this.onScrollerIn, this.onScrollerOut, this);
  464.             this.scroller.topRepeater = new Ext.util.ClickRepeater(this.scroller.top, {
  465.                 listeners: {
  466.                     click: this.onScroll.createDelegate(this, [null, this.scroller.top], false)
  467.                 }
  468.             });
  469.             this.scroller.bottom.hover(this.onScrollerIn, this.onScrollerOut, this);
  470.             this.scroller.bottomRepeater = new Ext.util.ClickRepeater(this.scroller.bottom, {
  471.                 listeners: {
  472.                     click: this.onScroll.createDelegate(this, [null, this.scroller.bottom], false)
  473.                 }
  474.             });
  475.         }
  476.     },
  477.     onLayout: function(){
  478.         if(this.isVisible()){
  479.             if(this.enableScrolling){
  480.                 this.constrainScroll(this.el.getTop());
  481.             }
  482.             if(this.floating){
  483.                 this.el.sync();
  484.             }
  485.         }
  486.     },
  487.     focus : function(){
  488.         if(!this.hidden){
  489.             this.doFocus.defer(50, this);
  490.         }
  491.     },
  492.     doFocus : function(){
  493.         if(!this.hidden){
  494.             this.focusEl.focus();
  495.         }
  496.     },
  497.     /**
  498.      * Hides this menu and optionally all parent menus
  499.      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
  500.      */
  501.     hide : function(deep){
  502.         this.deepHide = deep;
  503.         Ext.menu.Menu.superclass.hide.call(this);
  504.         delete this.deepHide;
  505.     },
  506.     // private
  507.     onHide: function(){
  508.         Ext.menu.Menu.superclass.onHide.call(this);
  509.         this.deactivateActive();
  510.         if(this.el && this.floating){
  511.             this.el.hide();
  512.         }
  513.         if(this.deepHide === true && this.parentMenu){
  514.             this.parentMenu.hide(true);
  515.         }
  516.     },
  517.     // private
  518.     lookupComponent: function(c){
  519.          if(Ext.isString(c)){
  520.             c = (c == 'separator' || c == '-') ? new Ext.menu.Separator() : new Ext.menu.TextItem(c);
  521.              this.applyDefaults(c);
  522.          }else{
  523.             if(Ext.isObject(c)){
  524.                 c = this.getMenuItem(c);
  525.             }else if(c.tagName || c.el){ // element. Wrap it.
  526.                 c = new Ext.BoxComponent({
  527.                     el: c
  528.                 });
  529.             }
  530.          }
  531.          return c;
  532.     },
  533.     applyDefaults : function(c){
  534.         if(!Ext.isString(c)){
  535.             c = Ext.menu.Menu.superclass.applyDefaults.call(this, c);
  536.             var d = this.internalDefaults;
  537.             if(d){
  538.                 if(c.events){
  539.                     Ext.applyIf(c.initialConfig, d);
  540.                     Ext.apply(c, d);
  541.                 }else{
  542.                     Ext.applyIf(c, d);
  543.                 }
  544.             }
  545.         }
  546.         return c;
  547.     },
  548.     // private
  549.     getMenuItem: function(config){
  550.        if(!config.isXType){
  551.             if(!config.xtype && Ext.isBoolean(config.checked)){
  552.                 return new Ext.menu.CheckItem(config)
  553.             }
  554.             return Ext.create(config, this.defaultType);
  555.         }
  556.         return config;
  557.     },
  558.     /**
  559.      * Adds a separator bar to the menu
  560.      * @return {Ext.menu.Item} The menu item that was added
  561.      */
  562.     addSeparator : function(){
  563.         return this.add(new Ext.menu.Separator());
  564.     },
  565.     /**
  566.      * Adds an {@link Ext.Element} object to the menu
  567.      * @param {Mixed} el The element or DOM node to add, or its id
  568.      * @return {Ext.menu.Item} The menu item that was added
  569.      */
  570.     addElement : function(el){
  571.         return this.add(new Ext.menu.BaseItem(el));
  572.     },
  573.     /**
  574.      * Adds an existing object based on {@link Ext.menu.BaseItem} to the menu
  575.      * @param {Ext.menu.Item} item The menu item to add
  576.      * @return {Ext.menu.Item} The menu item that was added
  577.      */
  578.     addItem : function(item){
  579.         return this.add(item);
  580.     },
  581.     /**
  582.      * Creates a new {@link Ext.menu.Item} based an the supplied config object and adds it to the menu
  583.      * @param {Object} config A MenuItem config object
  584.      * @return {Ext.menu.Item} The menu item that was added
  585.      */
  586.     addMenuItem : function(config){
  587.         return this.add(this.getMenuItem(config));
  588.     },
  589.     /**
  590.      * Creates a new {@link Ext.menu.TextItem} with the supplied text and adds it to the menu
  591.      * @param {String} text The text to display in the menu item
  592.      * @return {Ext.menu.Item} The menu item that was added
  593.      */
  594.     addText : function(text){
  595.         return this.add(new Ext.menu.TextItem(text));
  596.     },
  597.     //private
  598.     onDestroy : function(){
  599.         Ext.menu.Menu.superclass.onDestroy.call(this);
  600.         Ext.menu.MenuMgr.unregister(this);
  601.         Ext.EventManager.removeResizeListener(this.hide, this);
  602.         if(this.keyNav) {
  603.             this.keyNav.disable();
  604.         }
  605.         var s = this.scroller;
  606.         if(s){
  607.             Ext.destroy(s.topRepeater, s.bottomRepeater, s.top, s.bottom);
  608.         }
  609.     }
  610. });
  611. Ext.reg('menu', Ext.menu.Menu);
  612. // MenuNav is a private utility class used internally by the Menu
  613. Ext.menu.MenuNav = Ext.extend(Ext.KeyNav, function(){
  614.     function up(e, m){
  615.         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
  616.             m.tryActivate(m.items.length-1, -1);
  617.         }
  618.     }
  619.     function down(e, m){
  620.         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
  621.             m.tryActivate(0, 1);
  622.         }
  623.     }
  624.     return {
  625.         constructor: function(menu){
  626.             Ext.menu.MenuNav.superclass.constructor.call(this, menu.el);
  627.             this.scope = this.menu = menu;
  628.         },
  629.         doRelay : function(e, h){
  630.             var k = e.getKey();
  631. //          Keystrokes within a form Field (e.g.: down in a Combo) do not navigate. Allow only TAB
  632.             if (this.menu.activeItem && this.menu.activeItem.isFormField && k != e.TAB) {
  633.                 return false;
  634.             }
  635.             if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
  636.                 this.menu.tryActivate(0, 1);
  637.                 return false;
  638.             }
  639.             return h.call(this.scope || this, e, this.menu);
  640.         },
  641.         tab: function(e, m) {
  642.             e.stopEvent();
  643.             if (e.shiftKey) {
  644.                 up(e, m);
  645.             } else {
  646.                 down(e, m);
  647.             }
  648.         },
  649.         up : up,
  650.         down : down,
  651.         right : function(e, m){
  652.             if(m.activeItem){
  653.                 m.activeItem.expandMenu(true);
  654.             }
  655.         },
  656.         left : function(e, m){
  657.             m.hide();
  658.             if(m.parentMenu && m.parentMenu.activeItem){
  659.                 m.parentMenu.activeItem.activate();
  660.             }
  661.         },
  662.         enter : function(e, m){
  663.             if(m.activeItem){
  664.                 e.stopPropagation();
  665.                 m.activeItem.onClick(e);
  666.                 m.fireEvent("click", this, m.activeItem);
  667.                 return true;
  668.             }
  669.         }
  670.     };
  671. }());