123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556 |
- /*!
- * Angular Material Design
- * https://github.com/angular/material
- * @license MIT
- * v0.11.4
- */
- goog.provide('ng.material.components.menuBar');
- goog.require('ng.material.components.menu');
- goog.require('ng.material.core');
- /**
- * @ngdoc module
- * @name material.components.menu-bar
- */
- angular.module('material.components.menuBar', [
- 'material.core',
- 'material.components.menu'
- ]);
- angular
- .module('material.components.menuBar')
- .controller('MenuBarController', MenuBarController);
- var BOUND_MENU_METHODS = ['handleKeyDown', 'handleMenuHover', 'scheduleOpenHoveredMenu', 'cancelScheduledOpen'];
- /**
- * ngInject
- */
- function MenuBarController($scope, $element, $attrs, $mdConstant, $document, $mdUtil, $timeout) {
- this.$element = $element;
- this.$attrs = $attrs;
- this.$mdConstant = $mdConstant;
- this.$mdUtil = $mdUtil;
- this.$document = $document;
- this.$scope = $scope;
- this.$timeout = $timeout;
- var self = this;
- angular.forEach(BOUND_MENU_METHODS, function(methodName) {
- self[methodName] = angular.bind(self, self[methodName]);
- });
- }
- MenuBarController.$inject = ["$scope", "$element", "$attrs", "$mdConstant", "$document", "$mdUtil", "$timeout"];
- MenuBarController.prototype.init = function() {
- var $element = this.$element;
- var $mdUtil = this.$mdUtil;
- var $scope = this.$scope;
- var self = this;
- $element.on('keydown', this.handleKeyDown);
- this.parentToolbar = $mdUtil.getClosest($element, 'MD-TOOLBAR');
- $scope.$on('$mdMenuOpen', function(event, el) {
- if (self.getMenus().indexOf(el[0]) != -1) {
- $element[0].classList.add('md-open');
- el[0].classList.add('md-open');
- self.currentlyOpenMenu = el.controller('mdMenu');
- self.currentlyOpenMenu.registerContainerProxy(self.handleKeyDown);
- self.enableOpenOnHover();
- }
- });
- $scope.$on('$mdMenuClose', function(event, el, opts) {
- var rootMenus = self.getMenus();
- if (rootMenus.indexOf(el[0]) != -1) {
- $element[0].classList.remove('md-open');
- el[0].classList.remove('md-open');
- }
- if (opts.closeAll) {
- if ($element[0].contains(el[0])) {
- var parentMenu = el[0];
- while (parentMenu && rootMenus.indexOf(parentMenu) == -1) {
- parentMenu = $mdUtil.getClosest(parentMenu, 'MD-MENU', true);
- }
- if (parentMenu) {
- if (!opts.skipFocus) parentMenu.querySelector('button:not([disabled])').focus();
- self.currentlyOpenMenu = undefined;
- self.disableOpenOnHover();
- self.setKeyboardMode(true);
- }
- }
- }
- });
- angular
- .element(this.getMenus())
- .on('mouseenter', this.handleMenuHover);
- this.setKeyboardMode(true);
- };
- MenuBarController.prototype.setKeyboardMode = function(enabled) {
- if (enabled) this.$element[0].classList.add('md-keyboard-mode');
- else this.$element[0].classList.remove('md-keyboard-mode');
- };
- MenuBarController.prototype.enableOpenOnHover = function() {
- if (this.openOnHoverEnabled) return;
- this.openOnHoverEnabled = true;
- var parentToolbar;
- if (parentToolbar = this.parentToolbar) {
- parentToolbar.dataset.mdRestoreStyle = parentToolbar.getAttribute('style');
- parentToolbar.style.position = 'relative';
- parentToolbar.style.zIndex = 100;
- }
- };
- MenuBarController.prototype.handleMenuHover = function(e) {
- this.setKeyboardMode(false);
- if (this.openOnHoverEnabled) {
- this.scheduleOpenHoveredMenu(e);
- }
- };
- MenuBarController.prototype.disableOpenOnHover = function() {
- if (!this.openOnHoverEnabled) return;
- this.openOnHoverEnabled = false;
- var parentToolbar;
- if (parentToolbar = this.parentToolbar) {
- parentToolbar.setAttribute('style', parentToolbar.dataset.mdRestoreStyle || '');
- }
- };
- MenuBarController.prototype.scheduleOpenHoveredMenu = function(e) {
- var menuEl = angular.element(e.currentTarget);
- var menuCtrl = menuEl.controller('mdMenu');
- this.setKeyboardMode(false);
- this.scheduleOpenMenu(menuCtrl);
- };
- MenuBarController.prototype.scheduleOpenMenu = function(menuCtrl) {
- var self = this;
- var $timeout = this.$timeout;
- if (menuCtrl != self.currentlyOpenMenu) {
- $timeout.cancel(self.pendingMenuOpen);
- self.pendingMenuOpen = $timeout(function() {
- self.pendingMenuOpen = undefined;
- if (self.currentlyOpenMenu) {
- self.currentlyOpenMenu.close(true, { closeAll: true });
- }
- menuCtrl.open();
- }, 200, false);
- }
- };
- MenuBarController.prototype.handleKeyDown = function(e) {
- var keyCodes = this.$mdConstant.KEY_CODE;
- var currentMenu = this.currentlyOpenMenu;
- var wasOpen = currentMenu && currentMenu.isOpen;
- this.setKeyboardMode(true);
- var handled, newMenu, newMenuCtrl;
- switch (e.keyCode) {
- case keyCodes.DOWN_ARROW:
- if (currentMenu) {
- currentMenu.focusMenuContainer();
- } else {
- this.openFocusedMenu();
- }
- handled = true;
- break;
- case keyCodes.UP_ARROW:
- currentMenu && currentMenu.close();
- handled = true;
- break;
- case keyCodes.LEFT_ARROW:
- newMenu = this.focusMenu(-1);
- if (wasOpen) {
- newMenuCtrl = angular.element(newMenu).controller('mdMenu');
- this.scheduleOpenMenu(newMenuCtrl);
- }
- handled = true;
- break;
- case keyCodes.RIGHT_ARROW:
- newMenu = this.focusMenu(+1);
- if (wasOpen) {
- newMenuCtrl = angular.element(newMenu).controller('mdMenu');
- this.scheduleOpenMenu(newMenuCtrl);
- }
- handled = true;
- break;
- }
- if (handled) {
- e && e.preventDefault && e.preventDefault();
- e && e.stopImmediatePropagation && e.stopImmediatePropagation();
- }
- };
- MenuBarController.prototype.focusMenu = function(direction) {
- var menus = this.getMenus();
- var focusedIndex = this.getFocusedMenuIndex();
- if (focusedIndex == -1) { focusedIndex = this.getOpenMenuIndex(); }
- var changed = false;
- if (focusedIndex == -1) { focusedIndex = 0; }
- else if (
- direction < 0 && focusedIndex > 0 ||
- direction > 0 && focusedIndex < menus.length - direction
- ) {
- focusedIndex += direction;
- changed = true;
- }
- if (changed) {
- menus[focusedIndex].querySelector('button').focus();
- return menus[focusedIndex];
- }
- };
- MenuBarController.prototype.openFocusedMenu = function() {
- var menu = this.getFocusedMenu();
- menu && angular.element(menu).controller('mdMenu').open();
- };
- MenuBarController.prototype.getMenus = function() {
- var $element = this.$element;
- return this.$mdUtil.nodesToArray($element[0].children)
- .filter(function(el) { return el.nodeName == 'MD-MENU'; });
- };
- MenuBarController.prototype.getFocusedMenu = function() {
- return this.getMenus()[this.getFocusedMenuIndex()];
- };
- MenuBarController.prototype.getFocusedMenuIndex = function() {
- var $mdUtil = this.$mdUtil;
- var focusedEl = $mdUtil.getClosest(
- this.$document[0].activeElement,
- 'MD-MENU'
- );
- if (!focusedEl) return -1;
- var focusedIndex = this.getMenus().indexOf(focusedEl);
- return focusedIndex;
- };
- MenuBarController.prototype.getOpenMenuIndex = function() {
- var menus = this.getMenus();
- for (var i = 0; i < menus.length; ++i) {
- if (menus[i].classList.contains('md-open')) return i;
- }
- return -1;
- };
- /**
- * @ngdoc directive
- * @name mdMenuBar
- * @module material.components.menu-bar
- * @restrict E
- * @description
- *
- * Menu bars are containers that hold multiple menus. They change the behavior and appearence
- * of the `md-menu` directive to behave similar to an operating system provided menu.
- *
- * @usage
- * <hljs lang="html">
- * <md-menu-bar>
- * <md-menu>
- * <button ng-click="$mdOpenMenu()">
- * File
- * </button>
- * <md-menu-content>
- * <md-menu-item>
- * <md-button ng-click="ctrl.sampleAction('share', $event)">
- * Share...
- * </md-button>
- * </md-menu-item>
- * <md-menu-divider></md-menu-divider>
- * <md-menu-item>
- * <md-menu-item>
- * <md-menu>
- * <md-button ng-click="$mdOpenMenu()">New</md-button>
- * <md-menu-content>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Document', $event)">Document</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Spreadsheet', $event)">Spreadsheet</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Presentation', $event)">Presentation</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Form', $event)">Form</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Drawing', $event)">Drawing</md-button></md-menu-item>
- * </md-menu-content>
- * </md-menu>
- * </md-menu-item>
- * </md-menu-content>
- * </md-menu>
- * </md-menu-bar>
- * </hljs>
- *
- * ## Menu Bar Controls
- *
- * You may place `md-menu-items` that function as controls within menu bars.
- * There are two modes that are exposed via the `type` attribute of the `md-menu-item`.
- * `type="checkbox"` will function as a boolean control for the `ng-model` attribute of the
- * `md-menu-item`. `type="radio"` will function like a radio button, setting the `ngModel`
- * to the `string` value of the `value` attribute. If you need non-string values, you can use
- * `ng-value` to provide an expression (this is similar to how angular's native `input[type=radio]` works.
- *
- * <hljs lang="html">
- * <md-menu-bar>
- * <md-menu>
- * <button ng-click="$mdOpenMenu()">
- * Sample Menu
- * </button>
- * <md-menu-content>
- * <md-menu-item type="checkbox" ng-model="settings.allowChanges">Allow changes</md-menu-item>
- * <md-menu-divider></md-menu-divider>
- * <md-menu-item type="radio" ng-model="settings.mode" ng-value="1">Mode 1</md-menu-item>
- * <md-menu-item type="radio" ng-model="settings.mode" ng-value="1">Mode 2</md-menu-item>
- * <md-menu-item type="radio" ng-model="settings.mode" ng-value="1">Mode 3</md-menu-item>
- * </md-menu-content>
- * </md-menu>
- * </md-menu-bar>
- * </hljs>
- *
- *
- * ### Nesting Menus
- *
- * Menus may be nested within menu bars. This is commonly called cascading menus.
- * To nest a menu place the nested menu inside the content of the `md-menu-item`.
- * <hljs lang="html">
- * <md-menu-item>
- * <md-menu>
- * <button ng-click="$mdOpenMenu()">New</md-button>
- * <md-menu-content>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Document', $event)">Document</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Spreadsheet', $event)">Spreadsheet</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Presentation', $event)">Presentation</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Form', $event)">Form</md-button></md-menu-item>
- * <md-menu-item><md-button ng-click="ctrl.sampleAction('New Drawing', $event)">Drawing</md-button></md-menu-item>
- * </md-menu-content>
- * </md-menu>
- * </md-menu-item>
- * </hljs>
- *
- */
- angular
- .module('material.components.menuBar')
- .directive('mdMenuBar', MenuBarDirective);
- /**
- *
- * @ngInjdect
- */
- function MenuBarDirective($mdUtil, $mdTheming) {
- return {
- restrict: 'E',
- require: 'mdMenuBar',
- controller: 'MenuBarController',
- compile: function compile(templateEl, templateAttrs) {
- if (!templateAttrs.ariaRole) {
- templateEl[0].setAttribute('role', 'menubar');
- }
- angular.forEach(templateEl[0].children, function(menuEl) {
- if (menuEl.nodeName == 'MD-MENU') {
- if (!menuEl.hasAttribute('md-position-mode')) {
- menuEl.setAttribute('md-position-mode', 'left bottom');
- }
- menuEl.setAttribute('role', 'menu');
- var contentEls = $mdUtil.nodesToArray(menuEl.querySelectorAll('md-menu-content'));
- angular.forEach(contentEls, function(contentEl) {
- contentEl.classList.add('md-menu-bar-menu');
- contentEl.classList.add('md-dense');
- if (!contentEl.hasAttribute('width')) {
- contentEl.setAttribute('width', 5);
- }
- });
- }
- });
- return function postLink(scope, el, attrs, ctrl) {
- $mdTheming(scope, el);
- ctrl.init();
- };
- }
- };
- }
- MenuBarDirective.$inject = ["$mdUtil", "$mdTheming"];
- angular
- .module('material.components.menuBar')
- .directive('mdMenuDivider', MenuDividerDirective);
- function MenuDividerDirective() {
- return {
- restrict: 'E',
- compile: function(templateEl, templateAttrs) {
- if (!templateAttrs.role) {
- templateEl[0].setAttribute('role', 'separator');
- }
- }
- };
- }
- angular
- .module('material.components.menuBar')
- .controller('MenuItemController', MenuItemController);
- /**
- * ngInject
- */
- function MenuItemController($scope, $element, $attrs) {
- this.$element = $element;
- this.$attrs = $attrs;
- this.$scope = $scope;
- }
- MenuItemController.$inject = ["$scope", "$element", "$attrs"];
- MenuItemController.prototype.init = function(ngModel) {
- var $element = this.$element;
- var $attrs = this.$attrs;
- this.ngModel = ngModel;
- if ($attrs.type == 'checkbox' || $attrs.type == 'radio') {
- this.mode = $attrs.type;
- this.iconEl = $element[0].children[0];
- this.buttonEl = $element[0].children[1];
- if (ngModel) this.initClickListeners();
- }
- };
- MenuItemController.prototype.initClickListeners = function() {
- var ngModel = this.ngModel;
- var $scope = this.$scope;
- var $attrs = this.$attrs;
- var $element = this.$element;
- var mode = this.mode;
- this.handleClick = angular.bind(this, this.handleClick);
- var icon = this.iconEl
- var button = angular.element(this.buttonEl);
- var handleClick = this.handleClick;
- $attrs.$observe('disabled', setDisabled);
- setDisabled($attrs.disabled);
- ngModel.$render = function render() {
- if (isSelected()) {
- icon.style.display = '';
- $element.attr('aria-checked', 'true');
- } else {
- icon.style.display = 'none';
- $element.attr('aria-checked', 'false');
- }
- };
- $scope.$$postDigest(ngModel.$render);
- function isSelected() {
- if (mode == 'radio') {
- var val = $attrs.ngValue ? $scope.$eval($attrs.ngValue) : $attrs.value;
- return ngModel.$modelValue == val;
- } else {
- return ngModel.$modelValue;
- }
- }
- function setDisabled(disabled) {
- if (disabled) {
- button.off('click', handleClick);
- } else {
- button.on('click', handleClick);
- }
- }
- };
- MenuItemController.prototype.handleClick = function(e) {
- var mode = this.mode;
- var ngModel = this.ngModel;
- var $attrs = this.$attrs;
- var newVal;
- if (mode == 'checkbox') {
- newVal = !ngModel.$modelValue;
- } else if (mode == 'radio') {
- newVal = $attrs.ngValue ? this.$scope.$eval($attrs.ngValue) : $attrs.value;
- }
- ngModel.$setViewValue(newVal);
- ngModel.$render();
- };
- angular
- .module('material.components.menuBar')
- .directive('mdMenuItem', MenuItemDirective);
- /**
- *
- * @ngInjdect
- */
- function MenuItemDirective() {
- return {
- require: ['mdMenuItem', '?ngModel'],
- compile: function(templateEl, templateAttrs) {
- if (templateAttrs.type == 'checkbox' || templateAttrs.type == 'radio') {
- var text = templateEl[0].textContent;
- var buttonEl = angular.element('<md-button type="button"></md-button>');
- buttonEl.html(text);
- buttonEl.attr('tabindex', '0');
- templateEl.html('');
- templateEl.append(angular.element('<md-icon md-svg-icon="check"></md-icon>'));
- templateEl.append(buttonEl);
- templateEl[0].classList.add('md-indent');
- setDefault('role', (templateAttrs.type == 'checkbox') ? 'menuitemcheckbox' : 'menuitemradio');
- angular.forEach(['ng-disabled'], moveAttrToButton);
- } else {
- setDefault('role', 'menuitem');
- }
- return function(scope, el, attrs, ctrls) {
- var ctrl = ctrls[0];
- var ngModel = ctrls[1];
- ctrl.init(ngModel);
- };
- function setDefault(attr, val) {
- if (!templateEl[0].hasAttribute(attr)) {
- templateEl[0].setAttribute(attr, val);
- }
- }
- function moveAttrToButton(attr) {
- if (templateEl[0].hasAttribute(attr)) {
- var val = templateEl[0].getAttribute(attr);
- buttonEl[0].setAttribute(attr, val);
- templateEl[0].removeAttribute(attr);
- }
- }
- },
- controller: 'MenuItemController'
- };
- }
- ng.material.components.menuBar = angular.module("material.components.menuBar");
|