Created
June 3, 2015 06:21
-
-
Save emolr/8653d423814f291535f7 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| /* | |
| * angular-ui-bootstrap | |
| * http://angular-ui.github.io/bootstrap/ | |
| * Version: 0.13.0 - 2015-05-02 | |
| * License: MIT | |
| */ | |
| angular.module("ui.bootstrap", ["ui.bootstrap.tpls","ui.bootstrap.dropdown","ui.bootstrap.position"]); | |
| angular.module("ui.bootstrap.tpls", []); | |
| angular.module('ui.bootstrap.dropdown', ['ui.bootstrap.position']) | |
| .constant('dropdownConfig', { | |
| openClass: 'open' | |
| }) | |
| .service('dropdownService', ['$document', '$rootScope', function($document, $rootScope) { | |
| var openScope = null; | |
| this.open = function( dropdownScope ) { | |
| if ( !openScope ) { | |
| $document.bind('click', closeDropdown); | |
| $document.bind('keydown', escapeKeyBind); | |
| } | |
| if ( openScope && openScope !== dropdownScope ) { | |
| openScope.isOpen = false; | |
| } | |
| openScope = dropdownScope; | |
| }; | |
| this.close = function( dropdownScope ) { | |
| if ( openScope === dropdownScope ) { | |
| openScope = null; | |
| $document.unbind('click', closeDropdown); | |
| $document.unbind('keydown', escapeKeyBind); | |
| } | |
| }; | |
| var closeDropdown = function( evt ) { | |
| // This method may still be called during the same mouse event that | |
| // unbound this event handler. So check openScope before proceeding. | |
| if (!openScope) { return; } | |
| if( evt && openScope.getAutoClose() === 'disabled' ) { return ; } | |
| var toggleElement = openScope.getToggleElement(); | |
| if ( evt && toggleElement && toggleElement[0].contains(evt.target) ) { | |
| return; | |
| } | |
| var $element = openScope.getElement(); | |
| if( evt && openScope.getAutoClose() === 'outsideClick' && $element && $element[0].contains(evt.target) ) { | |
| return; | |
| } | |
| openScope.isOpen = false; | |
| if (!$rootScope.$$phase) { | |
| openScope.$apply(); | |
| } | |
| }; | |
| var escapeKeyBind = function( evt ) { | |
| if ( evt.which === 27 ) { | |
| openScope.focusToggleElement(); | |
| closeDropdown(); | |
| } | |
| }; | |
| }]) | |
| .controller('DropdownController', ['$scope', '$attrs', '$parse', 'dropdownConfig', 'dropdownService', '$animate', '$position', '$document', function($scope, $attrs, $parse, dropdownConfig, dropdownService, $animate, $position, $document) { | |
| var self = this, | |
| scope = $scope.$new(), // create a child scope so we are not polluting original one | |
| openClass = dropdownConfig.openClass, | |
| getIsOpen, | |
| setIsOpen = angular.noop, | |
| toggleInvoker = $attrs.onToggle ? $parse($attrs.onToggle) : angular.noop, | |
| appendToBody = false; | |
| this.init = function( element ) { | |
| self.$element = element; | |
| if ( $attrs.isOpen ) { | |
| getIsOpen = $parse($attrs.isOpen); | |
| setIsOpen = getIsOpen.assign; | |
| $scope.$watch(getIsOpen, function(value) { | |
| scope.isOpen = !!value; | |
| }); | |
| } | |
| appendToBody = angular.isDefined($attrs.dropdownAppendToBody); | |
| if ( appendToBody && self.dropdownMenu ) { | |
| $document.find('body').append( self.dropdownMenu ); | |
| element.on('$destroy', function handleDestroyEvent() { | |
| self.dropdownMenu.remove(); | |
| }); | |
| } | |
| }; | |
| this.toggle = function( open ) { | |
| return scope.isOpen = arguments.length ? !!open : !scope.isOpen; | |
| }; | |
| // Allow other directives to watch status | |
| this.isOpen = function() { | |
| return scope.isOpen; | |
| }; | |
| scope.getToggleElement = function() { | |
| return self.toggleElement; | |
| }; | |
| scope.getAutoClose = function() { | |
| return $attrs.autoClose || 'always'; //or 'outsideClick' or 'disabled' | |
| }; | |
| scope.getElement = function() { | |
| return self.$element; | |
| }; | |
| scope.focusToggleElement = function() { | |
| if ( self.toggleElement ) { | |
| self.toggleElement[0].focus(); | |
| } | |
| }; | |
| scope.$watch('isOpen', function( isOpen, wasOpen ) { | |
| if ( appendToBody && self.dropdownMenu ) { | |
| var pos = $position.positionElements(self.$element, self.dropdownMenu, 'bottom-left', true); | |
| self.dropdownMenu.css({ | |
| top: pos.top + 'px', | |
| left: pos.left + 'px', | |
| display: isOpen ? 'block' : 'none' | |
| }); | |
| } | |
| $animate[isOpen ? 'addClass' : 'removeClass'](self.$element, openClass); | |
| if ( isOpen ) { | |
| scope.focusToggleElement(); | |
| dropdownService.open( scope ); | |
| } else { | |
| dropdownService.close( scope ); | |
| } | |
| setIsOpen($scope, isOpen); | |
| if (angular.isDefined(isOpen) && isOpen !== wasOpen) { | |
| toggleInvoker($scope, { open: !!isOpen }); | |
| } | |
| }); | |
| $scope.$on('$locationChangeSuccess', function() { | |
| scope.isOpen = false; | |
| }); | |
| $scope.$on('$destroy', function() { | |
| scope.$destroy(); | |
| }); | |
| }]) | |
| .directive('dropdown', function() { | |
| return { | |
| controller: 'DropdownController', | |
| link: function(scope, element, attrs, dropdownCtrl) { | |
| dropdownCtrl.init( element ); | |
| } | |
| }; | |
| }) | |
| .directive('dropdownMenu', function() { | |
| return { | |
| restrict: 'AC', | |
| require: '?^dropdown', | |
| link: function(scope, element, attrs, dropdownCtrl) { | |
| if ( !dropdownCtrl ) { | |
| return; | |
| } | |
| dropdownCtrl.dropdownMenu = element; | |
| } | |
| }; | |
| }) | |
| .directive('dropdownToggle', function() { | |
| return { | |
| require: '?^dropdown', | |
| link: function(scope, element, attrs, dropdownCtrl) { | |
| if ( !dropdownCtrl ) { | |
| return; | |
| } | |
| dropdownCtrl.toggleElement = element; | |
| var toggleDropdown = function(event) { | |
| event.preventDefault(); | |
| if ( !element.hasClass('disabled') && !attrs.disabled ) { | |
| scope.$apply(function() { | |
| dropdownCtrl.toggle(); | |
| }); | |
| } | |
| }; | |
| element.bind('click', toggleDropdown); | |
| // WAI-ARIA | |
| element.attr({ 'aria-haspopup': true, 'aria-expanded': false }); | |
| scope.$watch(dropdownCtrl.isOpen, function( isOpen ) { | |
| element.attr('aria-expanded', !!isOpen); | |
| }); | |
| scope.$on('$destroy', function() { | |
| element.unbind('click', toggleDropdown); | |
| }); | |
| } | |
| }; | |
| }); | |
| angular.module('ui.bootstrap.position', []) | |
| /** | |
| * A set of utility methods that can be use to retrieve position of DOM elements. | |
| * It is meant to be used where we need to absolute-position DOM elements in | |
| * relation to other, existing elements (this is the case for tooltips, popovers, | |
| * typeahead suggestions etc.). | |
| */ | |
| .factory('$position', ['$document', '$window', function ($document, $window) { | |
| function getStyle(el, cssprop) { | |
| if (el.currentStyle) { //IE | |
| return el.currentStyle[cssprop]; | |
| } else if ($window.getComputedStyle) { | |
| return $window.getComputedStyle(el)[cssprop]; | |
| } | |
| // finally try and get inline style | |
| return el.style[cssprop]; | |
| } | |
| /** | |
| * Checks if a given element is statically positioned | |
| * @param element - raw DOM element | |
| */ | |
| function isStaticPositioned(element) { | |
| return (getStyle(element, 'position') || 'static' ) === 'static'; | |
| } | |
| /** | |
| * returns the closest, non-statically positioned parentOffset of a given element | |
| * @param element | |
| */ | |
| var parentOffsetEl = function (element) { | |
| var docDomEl = $document[0]; | |
| var offsetParent = element.offsetParent || docDomEl; | |
| while (offsetParent && offsetParent !== docDomEl && isStaticPositioned(offsetParent) ) { | |
| offsetParent = offsetParent.offsetParent; | |
| } | |
| return offsetParent || docDomEl; | |
| }; | |
| return { | |
| /** | |
| * Provides read-only equivalent of jQuery's position function: | |
| * http://api.jquery.com/position/ | |
| */ | |
| position: function (element) { | |
| var elBCR = this.offset(element); | |
| var offsetParentBCR = { top: 0, left: 0 }; | |
| var offsetParentEl = parentOffsetEl(element[0]); | |
| if (offsetParentEl != $document[0]) { | |
| offsetParentBCR = this.offset(angular.element(offsetParentEl)); | |
| offsetParentBCR.top += offsetParentEl.clientTop - offsetParentEl.scrollTop; | |
| offsetParentBCR.left += offsetParentEl.clientLeft - offsetParentEl.scrollLeft; | |
| } | |
| var boundingClientRect = element[0].getBoundingClientRect(); | |
| return { | |
| width: boundingClientRect.width || element.prop('offsetWidth'), | |
| height: boundingClientRect.height || element.prop('offsetHeight'), | |
| top: elBCR.top - offsetParentBCR.top, | |
| left: elBCR.left - offsetParentBCR.left | |
| }; | |
| }, | |
| /** | |
| * Provides read-only equivalent of jQuery's offset function: | |
| * http://api.jquery.com/offset/ | |
| */ | |
| offset: function (element) { | |
| var boundingClientRect = element[0].getBoundingClientRect(); | |
| return { | |
| width: boundingClientRect.width || element.prop('offsetWidth'), | |
| height: boundingClientRect.height || element.prop('offsetHeight'), | |
| top: boundingClientRect.top + ($window.pageYOffset || $document[0].documentElement.scrollTop), | |
| left: boundingClientRect.left + ($window.pageXOffset || $document[0].documentElement.scrollLeft) | |
| }; | |
| }, | |
| /** | |
| * Provides coordinates for the targetEl in relation to hostEl | |
| */ | |
| positionElements: function (hostEl, targetEl, positionStr, appendToBody) { | |
| var positionStrParts = positionStr.split('-'); | |
| var pos0 = positionStrParts[0], pos1 = positionStrParts[1] || 'center'; | |
| var hostElPos, | |
| targetElWidth, | |
| targetElHeight, | |
| targetElPos; | |
| hostElPos = appendToBody ? this.offset(hostEl) : this.position(hostEl); | |
| targetElWidth = targetEl.prop('offsetWidth'); | |
| targetElHeight = targetEl.prop('offsetHeight'); | |
| var shiftWidth = { | |
| center: function () { | |
| return hostElPos.left + hostElPos.width / 2 - targetElWidth / 2; | |
| }, | |
| left: function () { | |
| return hostElPos.left; | |
| }, | |
| right: function () { | |
| return hostElPos.left + hostElPos.width; | |
| } | |
| }; | |
| var shiftHeight = { | |
| center: function () { | |
| return hostElPos.top + hostElPos.height / 2 - targetElHeight / 2; | |
| }, | |
| top: function () { | |
| return hostElPos.top; | |
| }, | |
| bottom: function () { | |
| return hostElPos.top + hostElPos.height; | |
| } | |
| }; | |
| switch (pos0) { | |
| case 'right': | |
| targetElPos = { | |
| top: shiftHeight[pos1](), | |
| left: shiftWidth[pos0]() | |
| }; | |
| break; | |
| case 'left': | |
| targetElPos = { | |
| top: shiftHeight[pos1](), | |
| left: hostElPos.left - targetElWidth | |
| }; | |
| break; | |
| case 'bottom': | |
| targetElPos = { | |
| top: shiftHeight[pos0](), | |
| left: shiftWidth[pos1]() | |
| }; | |
| break; | |
| default: | |
| targetElPos = { | |
| top: hostElPos.top - targetElHeight, | |
| left: shiftWidth[pos1]() | |
| }; | |
| break; | |
| } | |
| return targetElPos; | |
| } | |
| }; | |
| }]); | |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment