diff options
Diffstat (limited to 'vnfmarket/src/main/webapp/common/thirdparty/angular-material/modules/js/menuBar/menuBar.js')
-rw-r--r-- | vnfmarket/src/main/webapp/common/thirdparty/angular-material/modules/js/menuBar/menuBar.js | 615 |
1 files changed, 615 insertions, 0 deletions
diff --git a/vnfmarket/src/main/webapp/common/thirdparty/angular-material/modules/js/menuBar/menuBar.js b/vnfmarket/src/main/webapp/common/thirdparty/angular-material/modules/js/menuBar/menuBar.js new file mode 100644 index 00000000..d2a10a45 --- /dev/null +++ b/vnfmarket/src/main/webapp/common/thirdparty/angular-material/modules/js/menuBar/menuBar.js @@ -0,0 +1,615 @@ +/*! + * Angular Material Design + * https://github.com/angular/material + * @license MIT + * v1.1.3 + */ +(function( window, angular, undefined ){ +"use strict"; + +/** + * @ngdoc module + * @name material.components.menuBar + */ + +angular.module('material.components.menuBar', [ + 'material.core', + 'material.components.icon', + 'material.components.menu' +]); + + +MenuBarController['$inject'] = ["$scope", "$rootScope", "$element", "$attrs", "$mdConstant", "$document", "$mdUtil", "$timeout"]; +angular + .module('material.components.menuBar') + .controller('MenuBarController', MenuBarController); + +var BOUND_MENU_METHODS = ['handleKeyDown', 'handleMenuHover', 'scheduleOpenHoveredMenu', 'cancelScheduledOpen']; + +/** + * ngInject + */ +function MenuBarController($scope, $rootScope, $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.$rootScope = $rootScope; + this.$timeout = $timeout; + + var self = this; + angular.forEach(BOUND_MENU_METHODS, function(methodName) { + self[methodName] = angular.bind(self, self[methodName]); + }); +} + +MenuBarController.prototype.init = function() { + var $element = this.$element; + var $mdUtil = this.$mdUtil; + var $scope = this.$scope; + + var self = this; + var deregisterFns = []; + $element.on('keydown', this.handleKeyDown); + this.parentToolbar = $mdUtil.getClosest($element, 'MD-TOOLBAR'); + + deregisterFns.push(this.$rootScope.$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(); + } + })); + + deregisterFns.push(this.$rootScope.$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 ($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); + } + } + })); + + $scope.$on('$destroy', function() { + self.disableOpenOnHover(); + while (deregisterFns.length) { + deregisterFns.shift()(); + } + }); + + + 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; + + var self = this; + + self.openOnHoverEnabled = true; + + if (self.parentToolbar) { + self.parentToolbar.classList.add('md-has-open-menu'); + + // Needs to be on the next tick so it doesn't close immediately. + self.$mdUtil.nextTick(function() { + angular.element(self.parentToolbar).on('click', self.handleParentClick); + }, false); + } + + angular + .element(self.getMenus()) + .on('mouseenter', self.handleMenuHover); +}; + +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; + + if (this.parentToolbar) { + this.parentToolbar.classList.remove('md-has-open-menu'); + angular.element(this.parentToolbar).off('click', this.handleParentClick); + } + + angular + .element(this.getMenus()) + .off('mouseenter', this.handleMenuHover); +}; + +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; changed = true; } + 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; +}; + +MenuBarController.prototype.handleParentClick = function(event) { + var openMenu = this.querySelector('md-menu.md-open'); + + if (openMenu && !openMenu.contains(event.target)) { + angular.element(openMenu).controller('mdMenu').close(true, { + closeAll: true + }); + } +}; + +/** + * @ngdoc directive + * @name mdMenuBar + * @module material.components.menuBar + * @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="$mdMenu.open()"> + * 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="$mdMenu.open()">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="$mdMenu.open()"> + * 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="$mdMenu.open()">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> + * + */ + +MenuBarDirective['$inject'] = ["$mdUtil", "$mdTheming"]; +angular + .module('material.components.menuBar') + .directive('mdMenuBar', MenuBarDirective); + +/* ngInject */ +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'); + + // Since we're in the compile function and actual `md-buttons` are not compiled yet, + // we need to query for possible `md-buttons` as well. + menuEl.querySelector('button, a, md-button').setAttribute('role', 'menuitem'); + } + 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); + } + }); + } + }); + + // Mark the child menu items that they're inside a menu bar. This is necessary, + // because mnMenuItem has special behaviour during compilation, depending on + // whether it is inside a mdMenuBar. We can usually figure this out via the DOM, + // however if a directive that uses documentFragment is applied to the child (e.g. ngRepeat), + // the element won't have a parent and won't compile properly. + templateEl.find('md-menu-item').addClass('md-in-menu-bar'); + + return function postLink(scope, el, attr, ctrl) { + el.addClass('_md'); // private md component indicator for styling + $mdTheming(scope, el); + ctrl.init(); + }; + } + }; + +} + + +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'); + } + } + }; +} + + +MenuItemController['$inject'] = ["$scope", "$element", "$attrs"]; +angular + .module('material.components.menuBar') + .controller('MenuItemController', MenuItemController); + + +/** + * ngInject + */ +function MenuItemController($scope, $element, $attrs) { + this.$element = $element; + this.$attrs = $attrs; + this.$scope = $scope; +} + +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) { + // Clear ngAria set attributes + this.initClickListeners(); + } + } +}; + +// ngAria auto sets attributes on a menu-item with a ngModel. +// We don't want this because our content (buttons) get the focus +// and set their own aria attributes appropritately. Having both +// breaks NVDA / JAWS. This undeoes ngAria's attrs. +MenuItemController.prototype.clearNgAria = function() { + var el = this.$element[0]; + var clearAttrs = ['role', 'tabindex', 'aria-invalid', 'aria-checked']; + angular.forEach(clearAttrs, function(attr) { + el.removeAttribute(attr); + }); +}; + +MenuItemController.prototype.initClickListeners = function() { + var self = this; + 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() { + self.clearNgAria(); + if (isSelected()) { + icon.style.display = ''; + button.attr('aria-checked', 'true'); + } else { + icon.style.display = 'none'; + button.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(); +}; + + +MenuItemDirective['$inject'] = ["$mdUtil", "$mdConstant", "$$mdSvgRegistry"]; +angular + .module('material.components.menuBar') + .directive('mdMenuItem', MenuItemDirective); + + /* ngInject */ +function MenuItemDirective($mdUtil, $mdConstant, $$mdSvgRegistry) { + return { + controller: 'MenuItemController', + require: ['mdMenuItem', '?ngModel'], + priority: $mdConstant.BEFORE_NG_ARIA, + compile: function(templateEl, templateAttrs) { + var type = templateAttrs.type; + var inMenuBarClass = 'md-in-menu-bar'; + + // Note: This allows us to show the `check` icon for the md-menu-bar items. + // The `md-in-menu-bar` class is set by the mdMenuBar directive. + if ((type == 'checkbox' || type == 'radio') && templateEl.hasClass(inMenuBarClass)) { + var text = templateEl[0].textContent; + var buttonEl = angular.element('<md-button type="button"></md-button>'); + var iconTemplate = '<md-icon md-svg-src="' + $$mdSvgRegistry.mdChecked + '"></md-icon>'; + + buttonEl.html(text); + buttonEl.attr('tabindex', '0'); + + templateEl.html(''); + templateEl.append(angular.element(iconTemplate)); + templateEl.append(buttonEl); + templateEl.addClass('md-indent').removeClass(inMenuBarClass); + + setDefault('role', type == 'checkbox' ? 'menuitemcheckbox' : 'menuitemradio', buttonEl); + moveAttrToButton('ng-disabled'); + + } else { + setDefault('role', 'menuitem', templateEl[0].querySelector('md-button, button, a')); + } + + + return function(scope, el, attrs, ctrls) { + var ctrl = ctrls[0]; + var ngModel = ctrls[1]; + ctrl.init(ngModel); + }; + + function setDefault(attr, val, el) { + el = el || templateEl; + if (el instanceof angular.element) { + el = el[0]; + } + if (!el.hasAttribute(attr)) { + el.setAttribute(attr, val); + } + } + + function moveAttrToButton(attribute) { + var attributes = $mdUtil.prefixer(attribute); + + angular.forEach(attributes, function(attr) { + if (templateEl[0].hasAttribute(attr)) { + var val = templateEl[0].getAttribute(attr); + buttonEl[0].setAttribute(attr, val); + templateEl[0].removeAttribute(attr); + } + }); + } + } + }; +} + +})(window, window.angular);
\ No newline at end of file |