/**
* @license
* Copyright The Closure Library Authors.
* SPDX-License-Identifier: Apache-2.0
*/
/**
* @fileoverview A class representing menu items that open a submenu.
* @see goog.ui.Menu
*
* @see ../demos/submenus.html
* @see ../demos/submenus2.html
*/
goog.provide('goog.ui.SubMenu');
goog.require('goog.Timer');
goog.require('goog.asserts');
goog.require('goog.dom');
goog.require('goog.dom.classlist');
goog.require('goog.events.KeyCodes');
goog.require('goog.positioning.AnchoredViewportPosition');
goog.require('goog.positioning.Corner');
goog.require('goog.style');
goog.require('goog.ui.Component');
goog.require('goog.ui.Menu');
goog.require('goog.ui.MenuItem');
goog.require('goog.ui.SubMenuRenderer');
goog.require('goog.ui.registry');
goog.requireType('goog.events.BrowserEvent');
goog.requireType('goog.events.Event');
goog.requireType('goog.events.KeyEvent');
goog.requireType('goog.ui.ControlContent');
goog.requireType('goog.ui.MenuHeader');
goog.requireType('goog.ui.MenuItemRenderer');
goog.requireType('goog.ui.MenuSeparator');
/**
* Class representing a submenu that can be added as an item to other menus.
*
* @param {goog.ui.ControlContent} content Text caption or DOM structure to
* display as the content of the submenu (use to add icons or styling to
* menus).
* @param {*=} opt_model Data/model associated with the menu item.
* @param {goog.dom.DomHelper=} opt_domHelper Optional dom helper used for dom
* interactions.
* @param {goog.ui.MenuItemRenderer=} opt_renderer Renderer used to render or
* decorate the component; defaults to {@link goog.ui.SubMenuRenderer}.
* @constructor
* @extends {goog.ui.MenuItem}
*/
goog.ui.SubMenu = function(content, opt_model, opt_domHelper, opt_renderer) {
'use strict';
goog.ui.MenuItem.call(
this, content, opt_model, opt_domHelper,
opt_renderer || goog.ui.SubMenuRenderer.getInstance());
};
goog.inherits(goog.ui.SubMenu, goog.ui.MenuItem);
/**
* The delay before opening the sub menu in milliseconds.
* @type {number}
*/
goog.ui.SubMenu.MENU_DELAY_MS = 218;
/**
* Timer used to dismiss the submenu when the item becomes unhighlighted.
* @type {?number}
* @private
*/
goog.ui.SubMenu.prototype.dismissTimer_ = null;
/**
* Timer used to show the submenu on mouseover.
* @type {?number}
* @private
*/
goog.ui.SubMenu.prototype.showTimer_ = null;
/**
* Whether the submenu believes the menu is visible.
* @type {boolean}
* @private
*/
goog.ui.SubMenu.prototype.menuIsVisible_ = false;
/**
* The lazily created sub menu.
* @type {goog.ui.Menu?}
* @private
*/
goog.ui.SubMenu.prototype.subMenu_ = null;
/**
* Whether or not the sub-menu was set explicitly.
* @type {boolean}
* @private
*/
goog.ui.SubMenu.prototype.externalSubMenu_ = false;
/**
* Whether or not to align the submenu at the end of the parent menu.
* If true, the menu expands to the right in LTR languages and to the left
* in RTL langauges.
* @type {boolean}
* @private
*/
goog.ui.SubMenu.prototype.alignToEnd_ = true;
/**
* Whether the position of this submenu may be adjusted to fit
* the visible area, as in {@link goog.ui.Popup.positionAtCoordinate}.
* @type {boolean}
* @private
*/
goog.ui.SubMenu.prototype.isPositionAdjustable_ = false;
/** @override */
goog.ui.SubMenu.prototype.enterDocument = function() {
'use strict';
goog.ui.SubMenu.superClass_.enterDocument.call(this);
this.getHandler().listen(
this.getParent(), goog.ui.Component.EventType.HIDE, this.onParentHidden_);
if (this.subMenu_) {
this.setMenuListenersEnabled_(this.subMenu_, true);
}
};
/** @override */
goog.ui.SubMenu.prototype.exitDocument = function() {
'use strict';
this.getHandler().unlisten(
this.getParent(), goog.ui.Component.EventType.HIDE, this.onParentHidden_);
if (this.subMenu_) {
this.setMenuListenersEnabled_(this.subMenu_, false);
if (!this.externalSubMenu_) {
this.subMenu_.exitDocument();
goog.dom.removeNode(this.subMenu_.getElement());
}
}
goog.ui.SubMenu.superClass_.exitDocument.call(this);
};
/** @override */
goog.ui.SubMenu.prototype.disposeInternal = function() {
'use strict';
if (this.subMenu_ && !this.externalSubMenu_) {
this.subMenu_.dispose();
}
this.subMenu_ = null;
goog.ui.SubMenu.superClass_.disposeInternal.call(this);
};
/**
* @override
* Dismisses the submenu on a delay, with the result that the user needs less
* accuracy when moving to submenus. Alternate implementations could use
* geometry instead of a timer.
* @param {boolean} highlight Whether item should be highlighted.
* @param {boolean=} opt_btnPressed Whether the mouse button is held down.
*/
goog.ui.SubMenu.prototype.setHighlighted = function(highlight, opt_btnPressed) {
'use strict';
goog.ui.SubMenu.superClass_.setHighlighted.call(this, highlight);
if (opt_btnPressed) {
this.getMenu().setMouseButtonPressed(true);
}
if (!highlight) {
if (this.dismissTimer_) {
goog.Timer.clear(this.dismissTimer_);
}
this.dismissTimer_ =
goog.Timer.callOnce(this.dismissSubMenu, this.getMenuDelay(), this);
}
};
/**
* Show the submenu and ensure that all siblings are hidden.
*/
goog.ui.SubMenu.prototype.showSubMenu = function() {
'use strict';
// Only show the menu if this item is still selected. This is called on a
// timeout, so make sure our parent still exists.
var parent = this.getParent();
if (parent && parent.getHighlighted() == this) {
this.setSubMenuVisible_(true);
this.dismissSiblings_();
}
};
/**
* Dismisses the menu and all further submenus.
*/
goog.ui.SubMenu.prototype.dismissSubMenu = function() {
'use strict';
// Because setHighlighted calls this function on a timeout, we need to make
// sure that the sub menu hasn't been disposed when we come back.
var subMenu = this.subMenu_;
if (subMenu && subMenu.getParent() == this) {
this.setSubMenuVisible_(false);
subMenu.forEachChild(function(child) {
'use strict';
if (typeof child.dismissSubMenu == 'function') {
child.dismissSubMenu();
}
});
}
};
/**
* Clears the show and hide timers for the sub menu.
*/
goog.ui.SubMenu.prototype.clearTimers = function() {
'use strict';
if (this.dismissTimer_) {
goog.Timer.clear(this.dismissTimer_);
}
if (this.showTimer_) {
goog.Timer.clear(this.showTimer_);
}
};
/**
* Sets the menu item to be visible or invisible.
* @param {boolean} visible Whether to show or hide the component.
* @param {boolean=} opt_force If true, doesn't check whether the component
* already has the requested visibility, and doesn't dispatch any events.
* @return {boolean} Whether the visibility was changed.
* @override
*/
goog.ui.SubMenu.prototype.setVisible = function(visible, opt_force) {
'use strict';
var visibilityChanged =
goog.ui.SubMenu.superClass_.setVisible.call(this, visible, opt_force);
// For menus that allow menu items to be hidden (i.e. ComboBox) ensure that
// the submenu is hidden.
if (visibilityChanged && !this.isVisible()) {
this.dismissSubMenu();
}
return visibilityChanged;
};
/**
* Dismiss all the sub menus of sibling menu items.
* @private
*/
goog.ui.SubMenu.prototype.dismissSiblings_ = function() {
'use strict';
this.getParent().forEachChild(function(child) {
'use strict';
if (child != this && typeof child.dismissSubMenu == 'function') {
child.dismissSubMenu();
child.clearTimers();
}
}, this);
};
/**
* Handles a key event that is passed to the menu item from its parent because
* it is highlighted. If the arrow keys or enter key is pressed the sub menu
* takes control and delegates further key events to its menu until it is
* dismissed.
* @param {goog.events.KeyEvent} e A key event.
* @return {boolean} Whether the event was handled.
* @override
*/
goog.ui.SubMenu.prototype.handleKeyEvent = function(e) {
'use strict';
var keyCode = e.keyCode;
var arrowOpenKeyCode = this.isRightToLeft() ? goog.events.KeyCodes.LEFT :
goog.events.KeyCodes.RIGHT;
var closeKeyCode = this.isRightToLeft() ? goog.events.KeyCodes.RIGHT :
goog.events.KeyCodes.LEFT;
if (!this.menuIsVisible_) {
// Menu item doesn't have keyboard control and the correct key was pressed.
// So open take keyboard control and open the sub menu.
if (this.isEnabled() &&
(keyCode == arrowOpenKeyCode || keyCode == goog.events.KeyCodes.ENTER ||
keyCode == this.getMnemonic())) {
this.showSubMenu();
this.getMenu().highlightFirst();
this.clearTimers();
// The menu item doesn't currently care about the key events so let the
// parent menu handle them accordingly .
} else {
return false;
}
// Menu item has control, so let its menu try to handle the keys (this may
// in turn be handled by sub-sub menus).
} else if (this.getMenu().handleKeyEvent(e)) {
// Nothing to do
// The menu has control and the key hasn't yet been handled, on left arrow
// we turn off key control.
} else if (keyCode == closeKeyCode) {
this.dismissSubMenu();
} else {
// Submenu didn't handle the key so let the parent decide what to do.
return false;
}
e.preventDefault();
return true;
};
/**
* Listens to the sub menus items and ensures that this menu item is selected
* while dismissing the others. This handles the case when the user mouses
* over other items on their way to the sub menu.
* @param {goog.events.Event} e Enter event to handle.
* @private
*/
goog.ui.SubMenu.prototype.onChildEnter_ = function(e) {
'use strict';
if (this.subMenu_.getParent() == this) {
this.clearTimers();
this.getParentEventTarget().setHighlighted(this);
this.dismissSiblings_();
}
};
/**
* Listens to the parent menu's hide event and ensures that all submenus are
* hidden at the same time.
* @param {goog.events.Event} e The event.
* @private
*/
goog.ui.SubMenu.prototype.onParentHidden_ = function(e) {
'use strict';
// Ignore propagated events
if (e.target == this.getParentEventTarget()) {
// TODO(user): Using an event for this is expensive. Consider having a
// generalized interface that the parent menu calls on its children when
// it is hidden.
this.dismissSubMenu();
this.clearTimers();
}
};
/**
* @override
* Sets a timer to show the submenu and then dispatches an ENTER event to the
* parent menu.
* @param {goog.events.BrowserEvent} e Mouse event to handle.
*/
goog.ui.SubMenu.prototype.handleMouseOver = function(e) {
'use strict';
if (this.isEnabled()) {
this.clearTimers();
this.showTimer_ =
goog.Timer.callOnce(this.showSubMenu, this.getMenuDelay(), this);
}
goog.ui.SubMenu.superClass_.handleMouseOver.call(this, e);
};
/**
* Returns the delay before opening or closing the menu in milliseconds.
* @return {number}
* @protected
*/
goog.ui.SubMenu.prototype.getMenuDelay = function() {
'use strict';
return goog.ui.SubMenu.MENU_DELAY_MS;
};
/**
* Overrides the default mouseup event handler, so that the ACTION isn't
* dispatched for the submenu itself, instead the submenu is shown instantly.
* @param {goog.events.Event} e The browser event.
* @return {boolean} True if the action was allowed to proceed, false otherwise.
* @override
*/
goog.ui.SubMenu.prototype.performActionInternal = function(e) {
'use strict';
this.clearTimers();
var shouldHandleClick =
this.isSupportedState(goog.ui.Component.State.SELECTED) ||
this.isSupportedState(goog.ui.Component.State.CHECKED);
if (shouldHandleClick) {
return goog.ui.SubMenu.superClass_.performActionInternal.call(this, e);
} else {
this.showSubMenu();
return true;
}
};
/**
* Sets the visiblility of the sub menu.
* @param {boolean} visible Whether to show menu.
* @private
*/
goog.ui.SubMenu.prototype.setSubMenuVisible_ = function(visible) {
'use strict';
// Unhighlighting the menuitems if closing the menu so the event handlers can
// determine the correct state.
if (!visible && this.getMenu()) {
this.getMenu().setHighlightedIndex(-1);
}
// Dispatch OPEN event before calling getMenu(), so we can create the menu
// lazily on first access.
this.dispatchEvent(
goog.ui.Component.getStateTransitionEvent(
goog.ui.Component.State.OPENED, visible));
var subMenu = this.getMenu();
if (visible != this.menuIsVisible_) {
goog.dom.classlist.enable(
goog.asserts.assert(this.getElement()),
goog.getCssName('goog-submenu-open'), visible);
}
if (visible != subMenu.isVisible()) {
if (visible) {
// Lazy-render menu when first shown, if needed.
if (!subMenu.isInDocument()) {
subMenu.render();
}
subMenu.setHighlightedIndex(-1);
}
subMenu.setVisible(visible);
// We must position after the menu is visible, otherwise positioning logic
// breaks in RTL.
if (visible) {
this.positionSubMenu();
}
}
this.menuIsVisible_ = visible;
};
/**
* Attaches or detaches menu event listeners to/from the given menu. Called
* each time a menu is attached to or detached from the submenu.
* @param {goog.ui.Menu} menu Menu on which to listen for events.
* @param {boolean} attach Whether to attach or detach event listeners.
* @private
*/
goog.ui.SubMenu.prototype.setMenuListenersEnabled_ = function(menu, attach) {
'use strict';
var handler = this.getHandler();
var method = attach ? handler.listen : handler.unlisten;
method.call(
handler, menu, goog.ui.Component.EventType.ENTER, this.onChildEnter_);
};
/**
* Sets whether the submenu is aligned at the end of the parent menu.
* @param {boolean} alignToEnd True to align to end, false to align to start.
*/
goog.ui.SubMenu.prototype.setAlignToEnd = function(alignToEnd) {
'use strict';
if (alignToEnd != this.alignToEnd_) {
this.alignToEnd_ = alignToEnd;
if (this.isInDocument()) {
// Completely re-render the widget.
var oldElement = this.getElement();
this.exitDocument();
if (oldElement.nextSibling) {
this.renderBefore(/** @type {!Element} */ (oldElement.nextSibling));
} else {
this.render(/** @type {Element} */ (oldElement.parentNode));
}
}
}
};
/**
* Determines whether the submenu is aligned at the end of the parent menu.
* @return {boolean} True if aligned to the end (the default), false if
* aligned to the start.
*/
goog.ui.SubMenu.prototype.isAlignedToEnd = function() {
'use strict';
return this.alignToEnd_;
};
/**
* Positions the submenu. This method should be called if the sub menu is
* opened and the menu element's size changes (e.g., when adding/removing items
* to an opened sub menu).
*/
goog.ui.SubMenu.prototype.positionSubMenu = function() {
'use strict';
var position = new goog.positioning.AnchoredViewportPosition(
this.getElement(),
this.isAlignedToEnd() ? goog.positioning.Corner.TOP_END :
goog.positioning.Corner.TOP_START,
this.isPositionAdjustable_);
// TODO(user): Clean up popup code and have this be a one line call
var subMenu = this.getMenu();
var el = subMenu.getElement();
if (!subMenu.isVisible()) {
el.style.visibility = 'hidden';
goog.style.setElementShown(el, true);
}
position.reposition(
el, this.isAlignedToEnd() ? goog.positioning.Corner.TOP_START :
goog.positioning.Corner.TOP_END);
if (!subMenu.isVisible()) {
goog.style.setElementShown(el, false);
el.style.visibility = 'visible';
}
};
// Methods delegated to sub-menu but accessible here for convinience
/**
* Adds a new menu item at the end of the menu.
* @param {goog.ui.MenuHeader|goog.ui.MenuItem|goog.ui.MenuSeparator} item Menu
* item to add to the menu.
*/
goog.ui.SubMenu.prototype.addItem = function(item) {
'use strict';
this.getMenu().addChild(item, true);
};
/**
* Adds a new menu item at a specific index in the menu.
* @param {goog.ui.MenuHeader|goog.ui.MenuItem|goog.ui.MenuSeparator} item Menu
* item to add to the menu.
* @param {number} n Index at which to insert the menu item.
*/
goog.ui.SubMenu.prototype.addItemAt = function(item, n) {
'use strict';
this.getMenu().addChildAt(item, n, true);
};
/**
* Removes an item from the menu and disposes it.
* @param {goog.ui.MenuItem} item The menu item to remove.
*/
goog.ui.SubMenu.prototype.removeItem = function(item) {
'use strict';
var child = this.getMenu().removeChild(item, true);
if (child) {
child.dispose();
}
};
/**
* Removes a menu item at a given index in the menu and disposes it.
* @param {number} n Index of item.
*/
goog.ui.SubMenu.prototype.removeItemAt = function(n) {
'use strict';
var child = this.getMenu().removeChildAt(n, true);
if (child) {
child.dispose();
}
};
/**
* Returns a reference to the menu item at a given index.
* @param {number} n Index of menu item.
* @return {goog.ui.Component} Reference to the menu item.
*/
goog.ui.SubMenu.prototype.getItemAt = function(n) {
'use strict';
return this.getMenu().getChildAt(n);
};
/**
* Returns the number of items in the sub menu (including separators).
* @return {number} The number of items in the menu.
*/
goog.ui.SubMenu.prototype.getItemCount = function() {
'use strict';
return this.getMenu().getChildCount();
};
/**
* Returns the menu items contained in the sub menu.
* @return {!Array<!goog.ui.MenuItem>} An array of menu items.
* @deprecated Use getItemAt/getItemCount instead.
*/
goog.ui.SubMenu.prototype.getItems = function() {
'use strict';
return this.getMenu().getItems();
};
/**
* Gets a reference to the submenu's actual menu.
* @return {!goog.ui.Menu} Reference to the object representing the sub menu.
*/
goog.ui.SubMenu.prototype.getMenu = function() {
'use strict';
if (!this.subMenu_) {
this.setMenu(
new goog.ui.Menu(this.getDomHelper()), /* opt_internal */ true);
} else if (this.externalSubMenu_ && this.subMenu_.getParent() != this) {
// Since it is possible for the same popup menu to be attached to multiple
// submenus, we need to ensure that it has the correct parent event target.
this.subMenu_.setParent(this);
}
// Always create the menu DOM, for backward compatibility.
if (!this.subMenu_.getElement()) {
this.subMenu_.createDom();
}
return this.subMenu_;
};
/**
* Sets the submenu to a specific menu.
* @param {goog.ui.Menu} menu The menu to show when this item is selected.
* @param {boolean=} opt_internal Whether this menu is an "internal" menu, and
* should be disposed of when this object is disposed of.
*/
goog.ui.SubMenu.prototype.setMenu = function(menu, opt_internal) {
'use strict';
var oldMenu = this.subMenu_;
if (menu != oldMenu) {
if (oldMenu) {
this.dismissSubMenu();
if (this.isInDocument()) {
this.setMenuListenersEnabled_(oldMenu, false);
}
}
this.subMenu_ = menu;
this.externalSubMenu_ = !opt_internal;
if (menu) {
menu.setParent(this);
// There's no need to dispatch a HIDE event during submenu construction.
menu.setVisible(false, /* opt_force */ true);
menu.setAllowAutoFocus(false);
menu.setFocusable(false);
if (this.isInDocument()) {
this.setMenuListenersEnabled_(menu, true);
}
}
}
};
/**
* Returns true if the provided element is to be considered inside the menu for
* purposes such as dismissing the menu on an event. This is so submenus can
* make use of elements outside their own DOM.
* @param {Element} element The element to test for.
* @return {boolean} Whether or not the provided element is contained.
*/
goog.ui.SubMenu.prototype.containsElement = function(element) {
'use strict';
return this.getMenu().containsElement(element);
};
/**
* @param {boolean} isAdjustable Whether this submenu is adjustable.
*/
goog.ui.SubMenu.prototype.setPositionAdjustable = function(isAdjustable) {
'use strict';
this.isPositionAdjustable_ = !!isAdjustable;
};
/**
* @return {boolean} Whether this submenu is adjustable.
*/
goog.ui.SubMenu.prototype.isPositionAdjustable = function() {
'use strict';
return this.isPositionAdjustable_;
};
// Register a decorator factory function for goog.ui.SubMenus.
goog.ui.registry.setDecoratorByClassName(
goog.getCssName('goog-submenu'), function() {
'use strict';
return new goog.ui.SubMenu(null);
});